Copepod Collection

Copepods were collected at approximately weekly intervals from Lake Champlain (Burlington Fishing Pier). Plankton was collected from the top 3 meters using a 250 um mesh net.

# # Lake Champlain near Burlington, VT
# siteNumber = "04294500"
# ChamplainInfo = readNWISsite(siteNumber)
# parameterCd = "00010"
# startDate = "2023-01-01"
# endDate = "2024-5-20"
# #statCd = c("00001", "00002","00003", "00011") # 1 - max, 2 - min, 3 = mean
# 
# # Constructs the URL for the data wanted then downloads the data
# url = constructNWISURL(siteNumbers = siteNumber, parameterCd = parameterCd,
#                        startDate = startDate, endDate = endDate, service = "uv")
# 
# raw_temps = importWaterML1(url, asDateTime = T) %>%
#   mutate("date" = as.Date(dateTime),
#          "hour" = hour(dateTime)) %>%
#   select(dateTime, tz_cd, date, hour, degC = X_00010_00000)
# 
# temp_data =  raw_temps %>%
#   select(date, hour, "temp" = degC)
# 
# write.csv(temp_data, file = "./Output/Data/champlain_temps.csv", row.names = F)

Collections began in late May 2023. Several gaps are present, but collections have continued at roughly weekly intervals since then. Copepods from 48 collections were used to make a total of 1312 thermal limit measurements. Over this time period, collection temperatures ranged from 2.5 to 26.5°C.

There is substantial variation in thermal limits across the species collected. There is also some degree of variation within the species, with thermal limits increasing slightly during the summer.

## Daily values for the period examined by dataset
collection_conditions = temp_data %>%
  ungroup() %>% 
  group_by(date) %>% 
  summarise(mean_temp = mean(temp),
            med_temp = median(temp),
            var_temp = var(temp), 
            min_temp = min(temp), 
            max_temp = max(temp)) %>% 
  mutate("range_temp" = max_temp - min_temp,
         date = as.Date(date)) %>% 
  ungroup() %>%  
  filter(date >= (min(as.Date(full_data$collection_date)) - 7)) %>% 
  left_join(unique(select(full_data, collection_date, collection_temp)), by = join_by(date == collection_date))

## Mean female thermal limits for each species, grouped by collection
species_summaries = full_data %>%  
  #filter(sex == "female") %>% 
  group_by(sp_name, collection_date, collection_temp) %>%  
  summarise("mean_ctmax" = mean(ctmax),
            "sample_size" = n(),
            "ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
            "ctmax_var" = var(ctmax), 
            "mean_size" = mean(size),
            "size_st_err" = (sd(size) / sqrt(sample_size)),
            "size_var" = var(size)) %>%  
  ungroup() %>% 
  complete(sp_name, collection_date) %>% 
  arrange(desc(sample_size))

adult_summaries = full_data %>%  
  filter(sex == "female") %>% 
  group_by(sp_name, collection_date, collection_temp) %>%  
  summarise("mean_ctmax" = mean(ctmax),
            "sample_size" = n(),
            "ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
            "ctmax_var" = var(ctmax), 
            "mean_size" = mean(size),
            "size_st_err" = (sd(size) / sqrt(sample_size)),
            "size_var" = var(size)) %>%  
  ungroup() %>% 
  complete(sp_name, collection_date) %>% 
  arrange(desc(sample_size))


tseries_data = full_data %>% 
  mutate(sp_name = fct_reorder(sp_name, ctmax, .desc = T))

ggplot() + 
  # geom_vline(data = unique(select(tseries_data, collection_date)), 
  #            aes(xintercept = as.Date(collection_date)),
  #            colour = "grey90",
  #            linewidth = 1) + 
  geom_line(data = collection_conditions, 
            aes(x = as.Date(date), y = mean_temp),
            colour = "black", 
            linewidth = 1) + 
  # geom_errorbar(data = species_summaries,
  #               aes(x = as.Date(collection_date),
  #                   ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
  #                   colour = sp_name),
  #               position = position_dodge(width = 1),
  #               width = 5, linewidth = 1) +
  # geom_point(data = adult_summaries, 
  #            aes(x = as.Date(collection_date), y = mean_ctmax, colour = sp_name, size = sample_size)) + 
  geom_point(data = tseries_data, 
             aes(x = as.Date(collection_date), y = ctmax, colour = sp_name),
             size = 1.5, shape = 16, alpha = 0.9,
             position = position_jitter(width = 0, height = 0)) + 
  scale_colour_manual(values = species_cols) + 
  guides(colour = guide_legend(override.aes = list(alpha = 1, size = 5))) + 
  labs(x = "Date", 
       y = "Temperature (°C)", 
       colour = "Species",
       size = "Sample Size") + 
  theme_matt() + 
  theme(legend.position = "right", 
        axis.text.x = element_text(angle = 320, hjust = 0, vjust = 0.5))


lake_temps = ggplot() + 
  geom_line(data = collection_conditions, 
            aes(x = as.Date(date), y = mean_temp),
            colour = "black", 
            linewidth = 1) + 
  labs(x = "Date", 
       y = "Temperature (°C)", 
       colour = "Species",
       size = "Sample Size") + 
  theme_matt() + 
  theme(legend.position = "right")

Temperatures observed at the time of collection closely resembled the maximum daily temperature from the temperature sensor data. Maximum temperature was used as a proxy instead of mean temperature as collections were usually made during afternoons or early evenings, just following the warmest part of the day.

collection_conditions %>% 
  drop_na(collection_temp) %>%  
  ggplot(aes(x = max_temp, y = collection_temp)) + 
  geom_abline(intercept = 0, slope = 1,
              linewidth = 1, colour = "grey") + 
  geom_point(size = 3) +
  scale_x_continuous(breaks = c(5,15,25)) + 
  scale_y_continuous(breaks = c(5,15,25)) + 
  labs(x = "Max. Temp. from Sensor (°C)",
       y = "Collection Temp. (°C)") + 
  theme_matt()

Size also varied, but primarily between rather than within species.

ggplot() + 
  geom_vline(data = unique(select(full_data, collection_date)), 
             aes(xintercept = as.Date(collection_date)),
             colour = "grey90",
             linewidth = 1) + 
  geom_line(data = collection_conditions, 
            aes(x = as.Date(date), y = mean_temp),
            colour = "black", 
            linewidth = 2) + 
  # geom_errorbar(data = species_summaries,
  #               aes(x = as.Date(collection_date), 
  #                   ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
  #                   colour = sp_name),
  #               position = position_dodge(width = 1),
  #               width = 5, linewidth = 1) + 
  geom_point(data = adult_summaries, 
             aes(x = as.Date(collection_date), y = mean_size * 40, colour = sp_name, size = sample_size),
             position = position_dodge(width = 1)) + 
  scale_colour_manual(values = species_cols) + 
  scale_y_continuous(
    name = "Temperature", # Features of the first axis
    sec.axis = sec_axis(~./40, name="Prosome Length (mm)"), # Add a second axis and specify its features
    breaks = c(0,5,10,15,20,25,30)
  ) + 
  labs(x = "Date", 
       y = "Temperature (°C)", 
       colour = "Species") + 
  theme_matt() + 
  theme(legend.position = "right")

sample_dates_plot = full_data %>%  
  filter(sp_name != "Osphranticum labronectum") %>% 
  mutate(sp_name = as.factor(sp_name),
         sp_name = fct_reorder(sp_name, ctmax)) %>% 
  ggplot(aes(x = lubridate::as_date(collection_date), 
             y = sp_name, fill = sp_name)) + 
  # geom_vline(xintercept = as_date(
  #   c("2023-05-01",
  #     "2023-09-01",
  #     "2024-01-01",
  #     "2024-05-01")),
  #   colour = "grey",
  #   linewidth = 1) + 
  geom_density_ridges(bandwidth = 30,
                      jittered_points = TRUE, 
                      point_shape = 21,
                      point_size = 1,
                      point_colour = "grey30",
                      point_alpha = 0.8,
                      alpha = 0.8,
                      position = position_points_jitter(
                        height = 0.1, width = 0)) + 
  scale_fill_manual(values = species_cols) + 
  scale_x_date(date_breaks = "3 months",
               date_labels = "%b") + 
  coord_cartesian(xlim = lubridate::as_date(c("2023-06-08", "2024-05-08"))) + 
  labs(x = "Day of Year", 
       y = "Species") + 
  theme_matt() + 
  #theme_ridges(grid = T) + 
  theme(legend.position = "none",
        axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))

The samples captured the broad seasonal changes in calanoid copepod community composition in the lake. We note, however, that rare species (e.g. Senecella and Limnocalanus) were often preferentially sampled, so are over-represented in the data set.

Throughout the season, the prevalence of various unidentified pathogens also varied, with very little infection observed during the Winter and Spring.

pathogen_cols = c("no" = "grey95", "cloudy" = "honeydew3", "spot" = "antiquewhite3", "other" = "tomato3")

full_data %>% 
  select(collection_date, dev_eggs, pathogen, lipids, sp_name, sex) %>% 
  group_by() %>% 
  filter(sex != "juvenile") %>% 
  group_by(collection_date) %>% 
  count(pathogen) %>% 
  filter(pathogen != "uncertain") %>% 
  pivot_wider(id_cols = "collection_date", 
              names_from = pathogen, 
              values_from = n,
              values_fill = 0) %>% 
  mutate(total = sum(no, cloudy, spot, other)) %>% 
  pivot_longer(cols = c(no, cloudy, spot, other),
               names_to = "pathogen", 
               values_to = "count") %>% 
  mutate(percent = count/total,
         collection_date = lubridate::as_date(collection_date),
         pathogen = fct_relevel(pathogen, "no", "cloudy", "spot", "other")) %>% 
  ggplot(aes(x = collection_date, y = percent, fill = pathogen)) + 
  geom_area() + 
  scale_fill_manual(values = pathogen_cols) + 
  scale_y_continuous(breaks = c(0,1)) + 
  labs(x = "Collection Date", 
       y = "Proportion", 
       fill = "Pathogen") + 
  theme_minimal(base_size = 20) + 
  theme(panel.grid = element_blank(),
        axis.ticks = element_line())

The transparent bodies of these copepods also allowed us to examine seasonal patterns in lipid reserves and in the production of eggs. Maturing oocytes are visible in female copepods before they are released. There was no strong seasonal cycle in the production of these eggs in any species, and instead, females were reproductively active throughout their respective seasons of occurence.

dev_eggs_cols = c("no" = "grey95", "yes" = "lightblue3")

full_data %>% 
  select(collection_date, dev_eggs, pathogen, lipids, sp_name, sex) %>% 
  group_by(sp_name) %>% 
  filter(sex != "juvenile") %>% 
  group_by(sp_name, collection_date) %>% 
  count(dev_eggs) %>% 
  filter(dev_eggs != "uncertain") %>% 
  pivot_wider(id_cols = c("collection_date", "sp_name"), 
              names_from = dev_eggs, 
              values_from = n,
              values_fill = 0) %>% 
  mutate(total = sum(no, yes)) %>% 
  pivot_longer(cols = c(no, yes),
               names_to = "dev_eggs", 
               values_to = "count") %>% 
  mutate(percent = count/total,
         collection_date = lubridate::as_date(collection_date),
         dev_eggs = fct_relevel(dev_eggs, "no", "yes")) %>% 
  ungroup() %>% 
  complete(collection_date, nesting(sp_name, dev_eggs), fill = list(percent = 1)) %>% 
  mutate(percent = if_else(is.na(total) & dev_eggs == "yes", 0, percent)) %>% 
  ggplot(aes(x = collection_date, y = percent, fill = dev_eggs)) + 
  facet_wrap(sp_name~., ncol = 1) + 
  geom_area() + 
  scale_fill_manual(values = dev_eggs_cols) + 
  scale_y_continuous(breaks = c(0,1)) + 
  labs(x = "Collection Date", 
       y = "Proportion", 
       fill = "Developing \nEggs") + 
  theme_minimal(base_size = 20) + 
  theme(panel.grid = element_blank(),
        axis.ticks = element_line())

The presence of lipids varied across species, with only L. minutus, L. sicilis, and Limnocalanus regularly possessing lipid stores.

lipid_cols = c("no" = "grey95", "yes" = "sienna2")

full_data %>% 
  select(collection_date, dev_eggs, pathogen, lipids, sp_name, sex) %>% 
  group_by(sp_name) %>% 
  filter(sex != "juvenile") %>% 
  group_by(sp_name, collection_date) %>% 
  count(lipids) %>% 
  filter(lipids != "uncertain") %>% 
  pivot_wider(id_cols = c("collection_date", "sp_name"), 
              names_from = lipids, 
              values_from = n,
              values_fill = 0) %>% 
  mutate(total = sum(no, yes)) %>% 
  pivot_longer(cols = c(no, yes),
               names_to = "lipids", 
               values_to = "count") %>% 
  mutate(percent = count/total,
         collection_date = lubridate::as_date(collection_date),
         lipids = fct_relevel(lipids, "no", "yes")) %>% 
  ungroup() %>% 
  complete(collection_date, nesting(sp_name, lipids), fill = list(percent = 1)) %>% 
  mutate(percent = if_else(is.na(total) & lipids == "yes", 0, percent)) %>% 
  ggplot(aes(x = collection_date, y = percent, fill = lipids)) + 
  facet_wrap(sp_name~., ncol = 1) + 
  geom_area() + 
  scale_fill_manual(values = lipid_cols) + 
  scale_y_continuous(breaks = c(0,1)) + 
  labs(x = "Collection Date", 
       y = "Proportion", 
       fill = "Lipids\nPresent") + 
  theme_minimal(base_size = 20) + 
  theme(panel.grid = element_blank(),
        axis.ticks = element_line())

Temperature Variability

Lake Champlain is highly seasonal, with both average temperatures and temperature variability changing throughout the year. These patterns in the experienced thermal environment may drive the observed variation in copepod thermal limits. However, the time period affecting copepod thermal limits is unknown. Depending the on the duration of time considered, there are large changes in the experienced environment, in particular regarding the temperature range and variance. Consider for example three time periods: the day of collection, one week prior to collection, and four weeks prior to collection. While the overall pattern is similar, we can see that, unsurprisingly, considering longer periods of time results in larger ranges and slightly changes the pattern of variance experienced.

## Defining the function to get predictor values for periods of different lengths
get_predictors = function(daily_values, raw_temp, n_days){
  prefix = str_replace_all(xfun::numbers_to_words(n_days), pattern = " ", replacement = "-")
  
  mean_values = daily_values %>% 
    ungroup() %>% 
    mutate(mean_max = slide_vec(.x = max_temp, .f = mean, .before = n_days, .complete = T),
           mean_min = slide_vec(.x = min_temp, .f = mean, .before = n_days, .complete = T),
           mean_range = slide_vec(.x = range_temp, .f = mean, .before = n_days, .complete = T)) %>% 
    select(date, mean_max, mean_min, mean_range) %>% 
    rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))
  
  period_values = raw_temp %>% 
    mutate(mean = slide_index_mean(temp, i = date, before = days(n_days), 
                                   na_rm = T),
           max = slide_index_max(temp, i = date, before = days(n_days), 
                                 na_rm = T),
           min = slide_index_min(temp, i = date, before = days(n_days),
                                 na_rm = T),
           med = slide_index_dbl(temp, .i = date, .before = days(n_days), 
                                 na_rm = T, .f = median),
           var = slide_index_dbl(temp, .i = date, .before = days(n_days), 
                                 .f = var),
           range = max - min) %>%  
    select(-temp) %>%  
    distinct() %>% 
    rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))%>% 
    inner_join(mean_values, by = c("date")) %>%  
    drop_na()
  
  return(period_values)
}

Organisms are unlikely to acclimate instantaneously to changes in temperature. To explore the potential temporal window these copepods are responding to, we examined the correlation between thermal limits and summaries of the thermal environment over different periods of time. For each species (inclusive of all sexes and stages), we examined the correlation between CTmax and one of nine representations of the thermal environment calculated for periods of time from 1 to 60 days before collection. These parameters include the overall maximum, minimum, median, and mean temperature for the period of time, the temperature range and variance during this time, and the mean daily temperature maximum, minimum, and range. We also examined the correlation between CTmax and the temperature recorded at the time of collection.

Shown below are the correlation coefficients for these relationships. Each facet shows the relationship for a different parameter, plotted against the duration of the time period before collection.

corr_vals %>% 
  mutate(parameter = fct_relevel(parameter, c("min", "max", "range",
                                              "mean", "med", "var",
                                              "mean_min", "mean_max", "mean_range"))) %>% 
  ggplot(aes(x = duration, y = correlation, colour = sp_name)) + 
  facet_wrap(.~parameter) + 
  geom_hline(yintercept = 0) + 
  geom_point(size = 0.9) + 
  geom_line(linewidth = 1.5) + 
  scale_colour_manual(values = species_cols) + 
  labs(x = "Duration (days)",
       y = "Correlation", 
       colour = "Species") + 
  theme_matt_facets()

This table contains the top three factors for each species (based on correlation coefficient).

Shown here is a graphical summary of the duration of the best predictors for each species. Note that for the two Leptodiaptomids, collection temperature had the largest correlation coefficient so duration is zero. This representation highlights that there is variation across the community not only in the potential driver (e.g. minimum vs. maximum temperatures) but also in the duration of time. This variation is not grouped by season (the winter and summer communities both have representative species apparently responding to short and long durations).

duration_plot = corr_vals %>%  
  filter(sig == "Sig.") %>% 
  drop_na(correlation) %>% 
  group_by(sp_name) %>%
  arrange(desc(correlation)) %>% 
  slice_head(n = 1) %>% 
  ungroup() %>% 
  mutate("num" = row_number(), 
         sp_name = fct_reorder(sp_name, duration, .fun = mean, .desc = T)) %>% 
  arrange(sp_name) %>% 
  select("Species" = sp_name, "Predictor" = parameter, "Duration" = duration, "Correlation" = correlation, num) %>% 
  ggplot(aes(x = Species, y = Duration, fill = Predictor, group = num)) + 
  geom_bar(stat = "identity", width = 0.5, position = position_dodge(width = 0.6),
           colour = "black") + 
  scale_fill_manual(values = c("coll_temp" = "black", "max" = "white", "min" = "grey")) + 
  labs(x = "", 
       y = "Duration \n(days)") + 
  theme_matt() + 
  theme(axis.text.x = element_blank())
correlation_coef_plot = corr_vals %>%  
  filter(sig == "Sig." | parameter == "coll_temp") %>% 
  drop_na(correlation) %>% 
  group_by(sp_name) %>%
  filter(parameter == "coll_temp" | correlation == max(correlation)) %>% 
  arrange(sp_name, parameter) %>% 
  mutate("num" = row_number()) %>% 
  select("Species" = sp_name, "Predictor" = parameter, "Duration" = duration, "Correlation" = correlation, num) %>% 
  mutate(Predictor = if_else(Predictor == "coll_temp", Predictor, "best")) %>% 
  ungroup() %>% 
  mutate(Species = fct_reorder(Species, Duration, .fun = max, .desc = T)) %>% 
  ggplot(aes(x = Species, y = Correlation, fill = Predictor, group = num)) + 
  geom_bar(stat = "identity", width = 0.5, position = position_dodge(width = 0.6),
           colour = "black") + 
  labs(y = "Correlation \nCoefficient",
       fill = "Correlate") +
  scale_fill_manual(values = c("coll_temp" = "black", "best" = "white")) + 
  scale_y_continuous(breaks = c(0, 1), limits = c(0,1)) +
  theme_matt() + 
  theme(axis.text.x = element_text(angle = 300, hjust = 0, vjust = 0.5))

ggarrange(duration_plot, correlation_coef_plot, nrow = 2, legend = "right", labels = "AUTO",
          heights = c(0.4, 0.6))

Trait Variation

Shown below are the clutch size distributions for the three diaptomiid species, which produce egg sacs that allow for easy quantification of fecundity.

full_data %>%  
  drop_na(fecundity) %>%  
  ggplot(aes(x = fecundity, fill = sp_name_sub)) + 
  facet_wrap(.~sp_name_sub, ncol = 1) + 
  geom_histogram(binwidth = 2) + 
  scale_fill_manual(values = species_cols) + 
  labs(x = "Fecundity (# Eggs)") +
  theme_matt_facets() + 
  theme(legend.position = "none")

One of the main aims of this project is to examine the patterns and processes driving variation in upper thermal limits across these species of copepods.

Variation with temperature

We expect one of the primary drivers of copepod thermal limits to be temperature, as individuals acclimate to seasonal changes. Shown below are the seasonal patterns of when copepods were included in CTmax measurements (a proxy for the season of occurrence), and thermal limits for each species plotted against the temperature at the time of collection. We generally see an increase in thermal limits with increasing collection temperature.

sp_ctmax_temp = full_data %>% 
  filter(sp_name != "Osphranticum labronectum") %>% 
  mutate(sp_name = as.factor(sp_name),
         sp_name = fct_reorder(sp_name, ctmax, .desc = T)) %>% 
  ggplot(aes(x = collection_temp, y = ctmax, colour = sp_name)) + 
  facet_wrap(sp_name~.) + 
  geom_smooth(method = "lm", se = F, linewidth = 1.5, colour = "grey30") + 
  geom_point(size = 2, alpha = 0.4) + 
  labs(x = "Collection Temp. (°C)", 
       y = "CTmax (°C)") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "none")

The interaction between seasonal changes in temperature and the acclimation of thermal limits likely affects vulnerability of each species to warming. Shown below are warming tolerance values for each species, calculated as the difference between individual CTmax and the temperature at the time of collection. All species maintained some degree of buffer between environmental temperatures and upper thermal limits, but L. minutus appears to approach its upper thermal limit during the warmest collections during the summer.

Also shown below is the relationship between fecundity (the number of eggs contained in a clutch) for the three diaptomid species. For the two Leptodiaptomus species, there is no relationship between clutch size and temperature, while there appears to be a general increase in clutch size with temperature in the Skistodiaptomus species.


wt_temp = full_data %>% 
  filter(sp_name != "Osphranticum labronectum") %>% 
  ggplot(aes(x = collection_temp, y = warming_tol, colour = sp_name)) + 
  geom_point(size = 3,
             alpha = 0.3) + 
  geom_smooth(method = "lm", linewidth = 3) +
  labs(x = "Collection Temperature (°C)", 
       y = "Warming Tolerance (°C)",
       colour = "Species")  + 
  ylim(0,30) + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "none")

eggs_temp = full_data %>% 
  filter(sp_name != "Osphranticum labronectum") %>% 
  ggplot(aes(x = collection_temp, y = fecundity, colour = sp_name)) + 
  geom_point(size = 3,
             alpha = 0.3) + 
  geom_smooth(method = "lm", linewidth = 3) +
  labs(x = "Collection Temperature (°C)", 
       y = "Fecundity (# Eggs)",
       colour = "Species")  + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "right")

ggarrange(wt_temp, eggs_temp, 
          common.legend = T, legend = "right", labels = "AUTO")

# Copepods spent several days in lab during experiments. Shown below are the CTmax residuals (taken from a model of CTmax against collection temperature) plotted against the time spent in lab before measurements were made. Individual regressions are shown for the residuals against days in lab for each collection. We can see clearly that thermal limits are fairly stable over time. 


# ggplot(ctmax_resids, aes(x = days_in_lab, y = resids, colour = sp_name, group = collection_date)) + 
#   facet_wrap(sp_name~.) + 
#   geom_point(size = 4, alpha = 0.5) + 
#   geom_smooth(method = "lm", se = F, linewidth = 1) + 
#   #scale_x_continuous(breaks = c(0:5)) + 
#   labs(x = "Days in lab", 
#        y = "CTmax Residuals") + 
#   scale_colour_manual(values = species_cols) + 
#   theme_matt_facets() + 
#   theme(legend.position = "none")

model_data = full_data %>%  
  drop_na(size, ctmax) %>%  
  filter(sp_name != "Osphranticum labronectum") %>% 
  mutate(temp_cent = scale(collection_temp, center = T, scale = F),
         size_cent = scale(size, center = T, scale = F))

minimal.model = lme4::lmer(data = model_data,
                           ctmax ~ sp_name + sex + temp_cent +
                             (1|days_in_lab))

full.model = lme4::lmer(data = model_data,
                        ctmax ~ sp_name*sex*temp_cent +
                          (1|days_in_lab))

drop1(full.model, test = "Chisq")
## Single term deletions
## 
## Model:
## ctmax ~ sp_name * sex * temp_cent + (1 | days_in_lab)
##                       npar    AIC    LRT Pr(Chi)  
## <none>                     5280.1                 
## sp_name:sex:temp_cent   10 5281.0 20.887  0.0219 *
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

performance::test_performance(minimal.model, full.model)
## Name          |   Model |    BF | df | df_diff |   Chi2 |      p
## ----------------------------------------------------------------
## minimal.model | lmerMod |       | 11 |         |        |       
## full.model    | lmerMod | 0.001 | 38 |   27.00 | 180.70 | < .001
## Models were detected as nested (in terms of fixed parameters) and are compared in sequential order.
performance::check_model(full.model)


car::Anova(full.model, type = "III")
## Analysis of Deviance Table (Type III Wald chisquare tests)
## 
## Response: ctmax
##                          Chisq Df Pr(>Chisq)    
## (Intercept)           4619.014  1  < 2.2e-16 ***
## sp_name                344.724  5  < 2.2e-16 ***
## sex                     51.233  2  7.495e-12 ***
## temp_cent               53.731  1  2.299e-13 ***
## sp_name:sex             10.611 10    0.38866    
## sp_name:temp_cent       39.512  5  1.873e-07 ***
## sex:temp_cent           31.961  2  1.147e-07 ***
## sp_name:sex:temp_cent   20.616 10    0.02393 *  
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

sp_ctmax = emmeans::emmeans(full.model, specs = "sp_name") %>% 
  data.frame() %>% 
  select(sp_name, "species_ctmax" = emmean)

model_coefs = emmeans::emtrends(full.model, var = "temp_cent", specs = "sp_name") %>% 
  data.frame() %>% 
  inner_join(sp_ctmax) 

ctmax_resids = model_data %>% 
  mutate(resids = residuals(full.model))

#write.csv(model_coefs, "Output/Data/ARR_data.csv")
arr_combined = synth_arr %>%
  filter(measure == "upper" & mean_lim > 20) %>% 
  select("group" = genus, arr, mean_lim) %>% 
  mutate("dataset" = "synthesis") %>% 
  bind_rows(
    select(model_coefs, "group" = sp_name, 'arr' = temp_cent.trend, 'mean_lim' = species_ctmax)
  ) %>% 
  mutate(dataset = if_else(is.na(dataset), "new data", "synthesis"),
         group = fct_reorder(group, arr, .desc = T))


ggplot(arr_combined, aes(x = mean_lim, y = arr)) + 
  geom_smooth(method = "lm", se = F, 
              linewidth = 2, colour = "grey30") + 
  geom_point(data = filter(arr_combined, dataset != "new data"), 
             size = 3, colour = "black", shape = 1, stroke = 1.4) + 
  geom_point(data = filter(arr_combined, dataset == "new data"),
             aes(colour = group), 
             size = 4.5) + 
  scale_colour_manual(values = species_cols) + 
  labs(x = "Thermal Limit", 
       y = "ARR", 
       colour = "Species") +
  theme_matt() + 
  theme(legend.position = "right")

Sex and stage variation in thermal limits

Previous sections have generally lumped juvenile, female, and male individuals together. There may be important stage- or sex-specific differences in CTmax though. For all species but Osphranticum, we have measurements for individuals in different stages and of different sexes.

sex_sample_sizes = full_data %>%  
  group_by(sp_name, sex) %>%  
  summarise(num = n()) %>%  
  pivot_wider(id_cols = sp_name,
              names_from = sex, 
              values_from = num,
              values_fill = 0) %>% 
  select("Species" = sp_name, "Juvenile" = juvenile, "Female" = female, "Male" = male)

knitr::kable(sex_sample_sizes, align = "c")
Species Juvenile Female Male
Epischura lacustris 37 45 20
Leptodiaptomus minutus 12 273 39
Leptodiaptomus sicilis 31 356 95
Limnocalanus macrurus 4 43 39
Osphranticum labronectum 0 1 0
Senecella calanoides 13 21 8
Skistodiaptomus sp 15 232 28

Across group comparisons show that there are generally no differences in thermal limits (represented here as the residuals from a CTmax ~ collection_temp x species linear regression), with the exception of Senecella males, which may have lower thermal limits (although sample sizes are very small in this group).

# ctmax_resids %>% 
#   filter(sp_name != "Osphranticum labronectum") %>% 
#   ggplot(aes(x = sex, y = resids, colour = sp_name)) + 
#   facet_wrap(sp_name~.) + 
#   geom_jitter(width = 0.1, alpha = 0.5) + 
#   geom_boxplot(width = 0.4, fill = NA, colour = "black", 
#                linewidth = 1, outlier.colour = NA) + 
#   scale_colour_manual(values = species_cols) + 
#   theme_matt_facets()
model2_data = model_data %>% 
  filter(sex == "female", 
         pathogen != "uncertain", 
         dev_eggs != "uncertain", 
         lipids != "uncertain") %>% 
  mutate(pathogens = fct_relevel(pathogen, "no", "spot", "cloudy", "other"))

other_factor_model = lmer(data = model2_data, 
                          ctmax~sp_name * collection_temp + dev_eggs + pathogen + lipids + (1|days_in_lab))

drop1(other_factor_model, scope = ~., test = "Chisq")
## Single term deletions
## 
## Model:
## ctmax ~ sp_name * collection_temp + dev_eggs + pathogen + lipids + 
##     (1 | days_in_lab)
##                         npar    AIC     LRT   Pr(Chi)    
## <none>                       3601.6                      
## sp_name                    5 3804.2 212.619 < 2.2e-16 ***
## collection_temp            0 3601.6   0.000              
## dev_eggs                   1 3602.0   2.391    0.1220    
## pathogen                   3 3629.4  33.832 2.149e-07 ***
## lipids                     1 3600.0   0.416    0.5189    
## sp_name:collection_temp    5 3635.7  44.139 2.171e-08 ***
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

reduced_factors_model = lmer(data = model2_data, 
                          ctmax~sp_name * collection_temp + pathogen + (1|days_in_lab))

performance::check_model(reduced_factors_model)


car::Anova(reduced_factors_model, type = "III")
## Analysis of Deviance Table (Type III Wald chisquare tests)
## 
## Response: ctmax
##                           Chisq Df Pr(>Chisq)    
## (Intercept)             570.328  1  < 2.2e-16 ***
## sp_name                 234.027  5  < 2.2e-16 ***
## collection_temp          72.589  1  < 2.2e-16 ***
## pathogen                 35.282  3  1.062e-07 ***
## sp_name:collection_temp  43.558  5  2.847e-08 ***
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
emmeans::emmeans(reduced_factors_model, spec = "pathogen") %>% emmeans::contrast(method="trt.vs.ctrl",ref="no") %>% plot() + 
  geom_vline(xintercept = 0) + 
  labs(x = "Difference (°C)", 
       y = "Comparison") + 
    theme_matt()

Trait Correlations and Trade-offs

A relationship between size and upper thermal limits has been suggested in a wide range of other taxa. Shown below are the measured upper thermal limits plotted against prosome length. The overall relationship (inclusive of all species) is shown as the black line in the background. Regressions for each individual species are also shown. Across the entire assemblage, there is a strong decrease in thermal limits with increasing size.

full_data %>% 
  #filter(sex == "female") %>%  
  ggplot( aes(x = size, y = ctmax, colour = sp_name)) + 
  geom_point(size = 2, alpha = 0.3) + 
  geom_smooth(data = full_data, 
              aes(x = size, y = ctmax),
              method = "lm", 
              colour ="black", 
              linewidth = 2.5) + 
  labs(x = "Length (mm)", 
       y = "CTmax (°C)",
       colour = "Species") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "right")

Shown here is the relationship for each species individually.

full_data %>% 
  #filter(sex == "female") %>%  
  group_by(sp_name) %>% filter(n() >2) %>% filter(!str_detect(sp_name, pattern = "kindti")) %>% 
  ggplot( aes(x = size, y = ctmax, colour = sp_name)) + 
  facet_wrap(sp_name~., scales = "free", nrow = 2) + 
  geom_point(size = 2, alpha = 0.8) + 
  geom_smooth(method = "lm", se = F, linewidth = 2) + 
  labs(x = "Length (mm)", 
       y = "CTmax (°C)",
       colour = "Species") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "none")

Shown below is the relationship between mean size and mean thermal limits for females of each species. We see that larger species within the community tend to have a lower thermal limit than smaller species.

full_data %>% 
  group_by(sp_name, sex) %>% 
  summarize(mean_ctmax = mean(ctmax, na.rm = T),
            mean_size = mean(size, na.rm = T)) %>% 
  #filter(sex == "female") %>% 
  ggplot(aes(x = mean_size, y = mean_ctmax)) + 
  geom_smooth(method = "lm", se = F, linewidth = 2, colour = "black") + 
  geom_point(aes(colour = sp_name, shape = sex),
             size = 5) + 
  labs(x = "Length (mm)", 
       y = "CTmax (°C)",
       colour = "Species") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "right")

Shown here is the relationship between fecundity and size, showing the classic pattern of increasing egg production with increasing size.

size_fecund_plot = ctmax_resids %>%  
  drop_na(fecundity) %>% 
  ggplot(aes(x = size, y = fecundity, colour = sp_name)) + 
  geom_smooth(method = "lm", se = F, linewidth = 2) + 
  geom_point(size = 2, alpha = 0.5) + 
  labs(x = "Prosome length (mm)", 
       y = "Fecundity (# Eggs)",
       colour = "Species") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "right")

Individuals may also allocate energy to different fitness related traits, prioritizing reproductive output over environmental tolerance, for example. Shown below is the relationship between CTmax residuals (again, controlling for the effects of collection temperature) against fecundity. We can see clearly that individuals with increased fecundity are not decreasing thermal limits, suggesting that there is no energetic trade-off between these traits.

ctmax_fecund_plot = ctmax_resids %>%  
  drop_na(fecundity) %>% 
  ggplot(aes(x = resids, y = fecundity, colour = sp_name)) + 
  geom_smooth(method = "lm", se = F, linewidth = 2) + 
  geom_point(size = 2, alpha = 0.5) + 
  labs(x = "CTmax Residuals (°C)", 
       y = "Fecundity (# Eggs)",
       colour = "Species") + 
  scale_colour_manual(values = species_cols) + 
  theme_matt() + 
  theme(legend.position = "right")

ggarrange(size_fecund_plot, ctmax_fecund_plot, ncol = 1, common.legend = T, labels = "AUTO", legend = "right")

Previous studies have shown that the magnitude of the size-fecundity correlation may be environmentally-dependent. This is not visible is the data from these collections.

corr_data = full_data %>% 
  drop_na(fecundity) %>% 
  filter(sp_name %in% c("Leptodiaptomus sicilis",
                        "Leptodiaptomus minutus", 
                        "Skistodiaptomus sp")) %>%  
  group_by(collection_date, collection_temp, sp_name) %>% 
  summarise(size_fec_corr = cor(size, fecundity),
            n = n(),
            mean_fecundity = mean(fecundity)) %>% 
  filter(n >= 3) %>% 
  ungroup() %>%  
  group_by(sp_name) %>% 
  mutate(temp_cent = scale(collection_temp, scale = F))

ggplot(corr_data, aes(x = temp_cent, y = size_fec_corr, colour = sp_name)) + 
  facet_wrap(sp_name~., nrow = 3) + 
  geom_hline(yintercept = 0) +
  geom_point(size = 3) + 
  geom_smooth(method = "lm", linewidth = 2) + 
  scale_color_manual(values = species_cols) + 
  labs(x = "Temperature (centered)", 
       y = "Correlation Coefficient") + 
  coord_cartesian(ylim = c(-1, 1)) +
  theme_matt_facets() + 
  theme(legend.position = "none")


# ggplot(corr_data, aes(x = size_fec_corr)) +
#     facet_wrap(sp_name~., nrow = 3) +
#   geom_histogram(binwidth = 0.2)

Other patterns in variation

Leptodiaptomus sicilis is the most abundant species during the winter. There was a large shift in the size of mature females towards the end of December. These large and small individuals are the same species (confirmed via COI sequencing), suggesting this shift may instead reflect a transition from one generation to another. This size difference may be caused by differences in the developmental environments. For example, individuals developing in January grow up at very low temperatures, and therefore may reach larger sizes. These individuals over-summer in deep waters, then re-emerge in October and produce a new generation. Water temperatures are still fairly high through November, which results in a generation of smaller individuals. These individuals mature in time to produce a new generation in January.

full_data %>%  
  filter(sp_name == "Leptodiaptomus sicilis") %>% 
  filter(sex != "juvenile") %>% 
  group_by(collection_date) %>% 
  mutate(size_center = scale(size, center = T, scale = F)) %>% 
  ggplot(aes(y = factor(collection_date), x = size, fill = collection_temp)) + 
  facet_wrap(sex~.) + 
  geom_density_ridges(bandwidth = 0.04) + 
  geom_vline(xintercept = 0.89) + 
  labs(x = "Size (mm)",
       y = "Date", 
       fill = "Coll. Temp. (°C)") + 
  theme_matt() + 
  theme(legend.position = "right",
        axis.text.y = element_text(size = 12))

A similar, but less distinct pattern can be observed in L. minutus individuals as well.

full_data %>%  
  filter(sp_name == "Leptodiaptomus minutus") %>% 
  filter(sex != "juvenile") %>% 
  ggplot(aes(y = factor(collection_date), x = size, fill = collection_temp)) + 
  facet_wrap(sex~.) + 
  geom_density_ridges(bandwidth = 0.04) + 
  geom_vline(xintercept = 0.69) + 
  labs(x = "Size (mm)",
       y = "Date", 
       fill = "Coll. Temp. (°C)") + 
  coord_cartesian(xlim = c(0.5,0.9)) + 
  theme_matt() + 
  theme(legend.position = "right",
        axis.text.y = element_text(size = 12))

Distribution Lag Non-Linear Model (DLNM approach)

Distributed lag models examine a response variable, measured at multiple time points, as a function of the lagged occurrence of some predictor variable (response y at time t as a function of predictor x(t-lag). This method utilizes a bi-dimensional dose-lag-response function, which essentially examines not only the dose effect, but the effect of the timing of the dose.

# Run this code, save the product, and then just read in the temp lag data object. Takes too long to run each time this document is knit. 

# lag_temps = temp_data %>%
#   group_by(date, hour) %>%
#   summarize("mean_temp" = mean(temp, na.rm = T)) %>%
#   ungroup() %>%
#   mutate(point_num = row_number())
# 
# uniq_days = length(unique(lag_temps$date))
# 
# g = gam(mean_temp ~ s(point_num, bs="cr", k=uniq_days + 10),
#     method = "REML",
#     data = lag_temps)
# 
# points = seq(1, nrow(lag_temps), length.out = length(lag_temps$hour))
# 
# df.res = df.residual(g)
# 
# pred_temps = predict(g, newdata = lag_temps, type = "response", se.fit = TRUE)
# 
# lag_temps = lag_temps %>%
#   mutate(trend_T = pred_temps$fit,
#          trend_se = pred_temps$se.fit,
#          temp_diff = mean_temp - trend_T)
# 
# write.csv(lag_temps, file = "./Output/Data/lag_temps.csv", row.names = F)

dlnm_data = full_data %>%  
  filter(sex == "female") %>% 
  filter(sp_name %in% c(
    "Leptodiaptomus sicilis",
    "Leptodiaptomus minutus"
  )) %>% 
  select(collection_date, collection_temp, sp_name, ctmax) %>% 
  group_by(collection_date, collection_temp, sp_name) %>%  
  summarise(mean_ctmax = mean(ctmax, na.rm = T),
            sample = n())

temp_data %>% 
  group_by(date) %>% 
  summarise(mean_temp = mean(temp),
            max_temp = max(temp, na.rm = T)) %>% 
  right_join(dlnm_data, by = join_by("date" == "collection_date")) %>% 
  ggplot(aes(x = max_temp, y = mean_ctmax)) + 
  facet_wrap(.~sp_name) + 
  geom_smooth(method = "gam") + 
  geom_point() + 
  labs(x = "Max Daily Temp. (°C)",
       y = "Mean CTmax (°C)") + 
  theme_matt_facets() + 
  theme(strip.text.x = element_text(size = 12))


sp_list = unique(dlnm_data$sp_name)

for(lag_species in sp_list){
  
  dlnm_data_sp = dlnm_data %>% 
    filter(sp_name == lag_species)
  
  # We need to estimate a matrix of exposure histories for each observation. This contains the series of exposures at each lag (l) for each of the n observations, constrained between l0 (minimum lag) and L (max lag). 
  
  dates = dlnm_data_sp$collection_date # For each of these dates, make a vector of the past 30 days (including the day of collection). NOTE: Don't use 'unique' dates here since some collections had multiple species
  
  exp_hist_z = data.frame()
  exp_hist_trend = data.frame()
  
  for(d in dates){
    
    history = lag_temps %>% 
      filter(date <= d & date > d - 10) %>% 
      arrange(desc(date), desc(hour)) %>% 
      mutate(lag = row_number() - 1) %>% 
      select(lag, mean_temp, temp_diff)
    
    z_vec = scale(history$mean_temp)[,1]
    names(z_vec) = history$lag
    
    trend_vec = history$temp_diff
    names(trend_vec) = history$lag
    
    exp_hist_z = bind_rows(exp_hist_z, z_vec)
    exp_hist_trend = bind_rows(exp_hist_trend, trend_vec)
    
  }
  
  #print(max(exp_hist_trend, na.rm = T))
  
  # The cross-basis function from dlnm will use the class of the x parameter to determine what to do. In our case, we need to provide it with the matrix of exposure histories for reach observation (row) and lag (column). 
  
  cb_temps = crossbasis(exp_hist_trend, lag = c(0,dim(exp_hist_trend)[2]-1), 
                        argvar =list(fun="cr",df=3), 
                        arglag=list(fun="cr",df=3,intercept=T))
  
  #summary(cb_temps)
  
  penalized_mat <- cbPen(cb_temps)
  
  #fitting GAM
  lag.gam = gam(data = dlnm_data_sp, 
                mean_ctmax ~ collection_temp + cb_temps, 
                method = "GCV.Cp",
                paraPen=list(cb_temps=penalized_mat))
  
  # summary(lag.gam)
  # AIC(lag.gam)
  
  #estimation of exposures effects
  
  #default plots
  pred_gam_Zs<-crosspred(cb_temps, lag.gam, 
                         cumul=F, cen=0, ci.level = 0.95,
                         at=seq(-4,4, 0.1))
  
  plot(pred_gam_Zs, "contour", main = lag_species, 
              xlab = "Temperature Deviation (°C)", 
              ylab = "Hours before collection")
  
  
  # 
  # plot(pred_gam_Zs, border = 2, cumul=F,
  #       theta=110,phi=20,ltheta=-80)
  
  # plot(pred_gam_Zs, "slices",
  #      var = c(3,-3),
  #      lag = c(1,200),
  #      col = 2)
  # 
}

Miscellany

run_starts = temp_record %>% 
  group_by(run) %>% 
  filter(minute_passed <= 3) %>% 
  summarise(start_temp = mean(temp_C)) %>% 
  mutate("temp_bin" = cut_number(start_temp, 4),
         temp_bin = case_when(
           temp_bin == "[2.7,9.26]" ~ "[2.7°C - 9.26°C]",
           temp_bin == "(9.26,13.9]" ~ "[9.26°C - 13.9°C]",
           temp_bin == "(13.9,21]" ~ "[13.9°C - 21°C]",
           temp_bin == "(21,30.5]" ~ "[21°C - 30.5°C]"
         ))

ramp_record2 = ramp_record %>% 
  group_by(run, minute_interval) %>% 
  summarise(mean_ramp = mean(ramp_per_minute)) %>% 
  ungroup() %>% 
  left_join(run_starts) %>% 
  mutate(temp_bin = fct_reorder(temp_bin, start_temp, .fun = mean))

ggplot(ramp_record2, aes(x = minute_interval, y = mean_ramp)) + 
  facet_wrap(temp_bin~.) + 
  geom_hline(yintercept = 0.3, colour = "grey") + 
  geom_hline(yintercept = 0.1, colour = "grey") + 
  geom_hex(aes(fill = cut(..count.., c(2, 5, 10, 20, 30, 40, 50))),
           bins = 30) + 
  scale_fill_viridis_d(name="Number of Observations",
                       labels = c("<5", "5-9", "10-19", "20-29", "30-39", "40-50"),
                       option = "mako") + 
  labs(y = "Ramp Rate (deg. C / min.)",
       x = "Time into run (minute)") + 
  theme_matt_facets(base_size = 12)


if(predict_vuln == F){

  knitr::knit_exit()

}
LS0tCnRpdGxlOiBTZWFzb25hbGl0eSBpbiBMYWtlIENoYW1wbGFpbiBDb3BlcG9kIFRoZXJtYWwgTGltaXRzCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCIKb3V0cHV0OiAKICBodG1sX2RvY3VtZW50OgogICAgICAgICAgY29kZV9mb2xkaW5nOiBoaWRlCiAgICAgICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCiAgICAgICAgICB0b2M6IHRydWUKICAgICAgICAgIHRvY19mbG9hdDogdHJ1ZQogIGdpdGh1Yl9kb2N1bWVudDoKICAgICAgICAgIGh0bWxfcHJldmlldzogZmFsc2UKICAgICAgICAgIHRvYzogdHJ1ZQogICAgICAgICAgdG9jX2RlcHRoOiAzCi0tLQoKYGBge3Igc2V0dXAsIGluY2x1ZGU9VCwgbWVzc2FnZSA9IEYsIHdhcm5pbmcgPSBGLCBlY2hvID0gRn0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KAogIGVjaG8gPSBrbml0cjo6aXNfaHRtbF9vdXRwdXQoKSwKICBmaWcuYWxpZ24gPSAiY2VudGVyIiwKICBmaWcucGF0aCA9ICIuLi9GaWd1cmVzL21hcmtkb3duLyIsCiAgZGV2ID0gYygicG5nIiwgInBkZiIpLAogIG1lc3NhZ2UgPSBGQUxTRSwKICB3YXJuaW5nID0gRkFMU0UsCiAgY29sbGFwc2UgPSBUCikKCnRoZW1lX21hdHQgPSBmdW5jdGlvbihiYXNlX3NpemUgPSAxOCwKICAgICAgICAgICAgICAgICAgICAgIGRhcmtfdGV4dCA9ICJncmV5MjAiKXsKICBtaWRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzJdCiAgbGlnaHRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzNdCiAgCiAgZ2dwdWJyOjp0aGVtZV9wdWJyKGJhc2VfZmFtaWx5PSJzYW5zIikgJStyZXBsYWNlJSAKICAgIHRoZW1lKAogICAgICBwYW5lbC5iYWNrZ3JvdW5kICA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksIAogICAgICBwbG90LmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLCAKICAgICAgbGVnZW5kLmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLAogICAgICBsZWdlbmQua2V5ID0gZWxlbWVudF9ibGFuaygpLAogICAgICB0ZXh0ID0gZWxlbWVudF90ZXh0KGNvbG91ciA9IG1pZF90ZXh0LCBsaW5laGVpZ2h0ID0gMS4xKSwKICAgICAgdGl0bGUgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuNSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3VyID0gZGFya190ZXh0KSwKICAgICAgYXhpcy50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBtaWRfdGV4dCksCiAgICAgIGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS4yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFyZ2luID0gdW5pdChjKDMsIDAsIDAsIDApLCAibW0iKSksCiAgICAgIGF4aXMudGl0bGUueSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS4yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFyZ2luID0gdW5pdChjKDAsIDUsIDAsIDApLCAibW0iKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbmdsZSA9IDkwKSwKICAgICAgbGVnZW5kLnRleHQgPSBlbGVtZW50X3RleHQoc2l6ZT1iYXNlX3NpemUgKiAwLjkpLAogICAgICBsZWdlbmQudGl0bGUgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDAuOSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmYWNlID0gImJvbGQiKSwKICAgICAgcGxvdC5tYXJnaW4gPSBtYXJnaW4oMC4yNSwgMC4yNSwgMC4yNSwgMC4yNSwiY20iKQogICAgKQp9Cgp0aGVtZV9tYXR0X2ZhY2V0cyA9IGZ1bmN0aW9uKGJhc2Vfc2l6ZSA9IDE4LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhcmtfdGV4dCA9ICJncmV5MjAiKXsKICBtaWRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzJdCiAgbGlnaHRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzNdCiAgCiAgdGhlbWVfYncoYmFzZV9mYW1pbHk9InNhbnMiKSAlK3JlcGxhY2UlIAogICAgdGhlbWUoCiAgICAgIHBhbmVsLmdyaWQgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgIHBhbmVsLmJhY2tncm91bmQgID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwgCiAgICAgIHBsb3QuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksIAogICAgICBsZWdlbmQuYmFja2dyb3VuZCA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksCiAgICAgIGxlZ2VuZC5rZXkgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLAogICAgICB0ZXh0ID0gZWxlbWVudF90ZXh0KGNvbG91ciA9IG1pZF90ZXh0LCBsaW5laGVpZ2h0ID0gMS4xKSwKICAgICAgc3RyaXAudGV4dC54ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUpLAogICAgICB0aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS41LAogICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBkYXJrX3RleHQpLAogICAgICBheGlzLnRleHQgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG91ciA9IG1pZF90ZXh0KSwKICAgICAgYXhpcy50aXRsZS54ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYXJnaW4gPSB1bml0KGMoMywgMCwgMCwgMCksICJtbSIpKSwKICAgICAgYXhpcy50aXRsZS55ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYXJnaW4gPSB1bml0KGMoMCwgNSwgMCwgMCksICJtbSIpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFuZ2xlID0gOTApLAogICAgICBsZWdlbmQudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplPWJhc2Vfc2l6ZSAqIDAuOSksCiAgICAgIGxlZ2VuZC50aXRsZSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMC45LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZhY2UgPSAiYm9sZCIpLAogICAgICBwbG90Lm1hcmdpbiA9IG1hcmdpbigwLjI1LCAwLjI1LCAwLjI1LCAwLjI1LCJjbSIpCiAgICApCn0KCnNwZWNpZXNfY29scyA9IGMoIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiID0gIiNmZmQwMjkiLAogICAgICAgICAgICAgICAgICJMZXB0b2RpYXB0b211cyBtaW51dHVzIGp1dmVuaWxlIiA9ICIjZTNkOGFmIiwKICAgICAgICAgICAgICAgICAiTGVwdG9kaWFwdG9tdXMgbWludXR1cyBtYWxlIiA9ICIjZmZlODk2IiwKICAgICAgICAgICAgICAgICAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIgPSAiI0Q4NkYyOSIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMgbWFsZSIgPSAiI0UyOEMwMCIsCiAgICAgICAgICAgICAgICAgIlNraXN0b2RpYXB0b211cyBzcCIgPSAiI0M1QzM1QSIsCiAgICAgICAgICAgICAgICAgIlNraXN0b2RpYXB0b211cyBzcCBtYWxlIiA9ICIjZDlkODkzIiwgCiAgICAgICAgICAgICAgICAgIlNraXN0b2RpYXB0b211cyBzcCBqdXZlbmlsZSIgPSAiI2U2ZTZhYSIsIAogICAgICAgICAgICAgICAgICJFcGlzY2h1cmEgbGFjdXN0cmlzIGp1dmVuaWxlIiA9ICJwbHVtMSIsIAogICAgICAgICAgICAgICAgICJFcGlzY2h1cmEgbGFjdXN0cmlzIG1hbGUiID0gInBsdW0zIiwgCiAgICAgICAgICAgICAgICAgIkVwaXNjaHVyYSBsYWN1c3RyaXMiID0gInBsdW00IiwgCiAgICAgICAgICAgICAgICAgIkxpbW5vY2FsYW51cyBtYWNydXJ1cyIgPSAic2t5Ymx1ZTQiLCAKICAgICAgICAgICAgICAgICAiTGltbm9jYWxhbnVzIG1hY3J1cnVzIG1hbGUiID0gInNreWJsdWUzIiwgCiAgICAgICAgICAgICAgICAgIkxpbW5vY2FsYW51cyBtYWNydXJ1cyBqdXZlbmlsZSIgPSAic2t5Ymx1ZSIsIAogICAgICAgICAgICAgICAgICJTZW5lY2VsbGEgY2FsYW5vaWRlcyIgPSAiZGFya3NlYWdyZWVuMyIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZG9yYSBraW5kdGkgbWFsZSIgPSAibGlnaHRibHVlMyIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZG9yYSBraW5kdGkiID0gImxpZ2h0Ymx1ZTQiLAogICAgICAgICAgICAgICAgICJMZXB0b2RvcmEga2luZHRpIGp1dmVuaWxlIiA9ICJsaWdodGJsdWUiLAogICAgICAgICAgICAgICAgICJPc3BocmFudGljdW0gbGFicm9uZWN0dW0iID0gInBpbmsiKQpgYGAKCiMjIENvcGVwb2QgQ29sbGVjdGlvbgoKQ29wZXBvZHMgd2VyZSBjb2xsZWN0ZWQgYXQgYXBwcm94aW1hdGVseSB3ZWVrbHkgaW50ZXJ2YWxzIGZyb20gTGFrZSBDaGFtcGxhaW4gKEJ1cmxpbmd0b24gRmlzaGluZyBQaWVyKS4gUGxhbmt0b24gd2FzIGNvbGxlY3RlZCBmcm9tIHRoZSB0b3AgMyBtZXRlcnMgdXNpbmcgYSAyNTAgdW0gbWVzaCBuZXQuIAoKYGBge3J9CiMgIyBMYWtlIENoYW1wbGFpbiBuZWFyIEJ1cmxpbmd0b24sIFZUCiMgc2l0ZU51bWJlciA9ICIwNDI5NDUwMCIKIyBDaGFtcGxhaW5JbmZvID0gcmVhZE5XSVNzaXRlKHNpdGVOdW1iZXIpCiMgcGFyYW1ldGVyQ2QgPSAiMDAwMTAiCiMgc3RhcnREYXRlID0gIjIwMjMtMDEtMDEiCiMgZW5kRGF0ZSA9ICIyMDI0LTUtMjAiCiMgI3N0YXRDZCA9IGMoIjAwMDAxIiwgIjAwMDAyIiwiMDAwMDMiLCAiMDAwMTEiKSAjIDEgLSBtYXgsIDIgLSBtaW4sIDMgPSBtZWFuCiMgCiMgIyBDb25zdHJ1Y3RzIHRoZSBVUkwgZm9yIHRoZSBkYXRhIHdhbnRlZCB0aGVuIGRvd25sb2FkcyB0aGUgZGF0YQojIHVybCA9IGNvbnN0cnVjdE5XSVNVUkwoc2l0ZU51bWJlcnMgPSBzaXRlTnVtYmVyLCBwYXJhbWV0ZXJDZCA9IHBhcmFtZXRlckNkLAojICAgICAgICAgICAgICAgICAgICAgICAgc3RhcnREYXRlID0gc3RhcnREYXRlLCBlbmREYXRlID0gZW5kRGF0ZSwgc2VydmljZSA9ICJ1diIpCiMgCiMgcmF3X3RlbXBzID0gaW1wb3J0V2F0ZXJNTDEodXJsLCBhc0RhdGVUaW1lID0gVCkgJT4lCiMgICBtdXRhdGUoImRhdGUiID0gYXMuRGF0ZShkYXRlVGltZSksCiMgICAgICAgICAgImhvdXIiID0gaG91cihkYXRlVGltZSkpICU+JQojICAgc2VsZWN0KGRhdGVUaW1lLCB0el9jZCwgZGF0ZSwgaG91ciwgZGVnQyA9IFhfMDAwMTBfMDAwMDApCiMgCiMgdGVtcF9kYXRhID0gIHJhd190ZW1wcyAlPiUKIyAgIHNlbGVjdChkYXRlLCBob3VyLCAidGVtcCIgPSBkZWdDKQojIAojIHdyaXRlLmNzdih0ZW1wX2RhdGEsIGZpbGUgPSAiLi9PdXRwdXQvRGF0YS9jaGFtcGxhaW5fdGVtcHMuY3N2Iiwgcm93Lm5hbWVzID0gRikKYGBgCgpDb2xsZWN0aW9ucyBiZWdhbiBpbiBsYXRlIE1heSAyMDIzLiBTZXZlcmFsIGdhcHMgYXJlIHByZXNlbnQsIGJ1dCBjb2xsZWN0aW9ucyBoYXZlIGNvbnRpbnVlZCBhdCByb3VnaGx5IHdlZWtseSBpbnRlcnZhbHMgc2luY2UgdGhlbi4gQ29wZXBvZHMgZnJvbSBgciBsZW5ndGgodW5pcXVlKGZ1bGxfZGF0YSRjb2xsZWN0aW9uX2RhdGUpKWAgY29sbGVjdGlvbnMgd2VyZSB1c2VkIHRvIG1ha2UgYSB0b3RhbCBvZiBgciBkaW0oZnVsbF9kYXRhKVsxXWAgdGhlcm1hbCBsaW1pdCBtZWFzdXJlbWVudHMuIE92ZXIgdGhpcyB0aW1lIHBlcmlvZCwgY29sbGVjdGlvbiB0ZW1wZXJhdHVyZXMgcmFuZ2VkIGZyb20gYHIgcGFzdGUobWluKGZ1bGxfZGF0YSRjb2xsZWN0aW9uX3RlbXApLCAiIHRvICIsIG1heChmdWxsX2RhdGEkY29sbGVjdGlvbl90ZW1wKSwgc2VwID0gIiIpYMKwQy4gICAgIAoKVGhlcmUgaXMgc3Vic3RhbnRpYWwgdmFyaWF0aW9uIGluIHRoZXJtYWwgbGltaXRzIGFjcm9zcyB0aGUgc3BlY2llcyBjb2xsZWN0ZWQuIFRoZXJlIGlzIGFsc28gc29tZSBkZWdyZWUgb2YgdmFyaWF0aW9uIHdpdGhpbiB0aGUgc3BlY2llcywgd2l0aCB0aGVybWFsIGxpbWl0cyBpbmNyZWFzaW5nIHNsaWdodGx5IGR1cmluZyB0aGUgc3VtbWVyLiAgICAKCmBgYHtyIG1haW4tZmlnLWN0bWF4LXRpbWVzZXJpZXMsIGZpZy53aWR0aD05LCBmaWcuaGVpZ2h0PTV9CiMjIERhaWx5IHZhbHVlcyBmb3IgdGhlIHBlcmlvZCBleGFtaW5lZCBieSBkYXRhc2V0CmNvbGxlY3Rpb25fY29uZGl0aW9ucyA9IHRlbXBfZGF0YSAlPiUKICB1bmdyb3VwKCkgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpc2UobWVhbl90ZW1wID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgbWVkX3RlbXAgPSBtZWRpYW4odGVtcCksCiAgICAgICAgICAgIHZhcl90ZW1wID0gdmFyKHRlbXApLCAKICAgICAgICAgICAgbWluX3RlbXAgPSBtaW4odGVtcCksIAogICAgICAgICAgICBtYXhfdGVtcCA9IG1heCh0ZW1wKSkgJT4lIAogIG11dGF0ZSgicmFuZ2VfdGVtcCIgPSBtYXhfdGVtcCAtIG1pbl90ZW1wLAogICAgICAgICBkYXRlID0gYXMuRGF0ZShkYXRlKSkgJT4lIAogIHVuZ3JvdXAoKSAlPiUgIAogIGZpbHRlcihkYXRlID49IChtaW4oYXMuRGF0ZShmdWxsX2RhdGEkY29sbGVjdGlvbl9kYXRlKSkgLSA3KSkgJT4lIAogIGxlZnRfam9pbih1bmlxdWUoc2VsZWN0KGZ1bGxfZGF0YSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApKSwgYnkgPSBqb2luX2J5KGRhdGUgPT0gY29sbGVjdGlvbl9kYXRlKSkKCiMjIE1lYW4gZmVtYWxlIHRoZXJtYWwgbGltaXRzIGZvciBlYWNoIHNwZWNpZXMsIGdyb3VwZWQgYnkgY29sbGVjdGlvbgpzcGVjaWVzX3N1bW1hcmllcyA9IGZ1bGxfZGF0YSAlPiUgIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2N0bWF4IiA9IG1lYW4oY3RtYXgpLAogICAgICAgICAgICAic2FtcGxlX3NpemUiID0gbigpLAogICAgICAgICAgICAiY3RtYXhfc3RfZXJyIiA9IChzZChjdG1heCkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJjdG1heF92YXIiID0gdmFyKGN0bWF4KSwgCiAgICAgICAgICAgICJtZWFuX3NpemUiID0gbWVhbihzaXplKSwKICAgICAgICAgICAgInNpemVfc3RfZXJyIiA9IChzZChzaXplKSAvIHNxcnQoc2FtcGxlX3NpemUpKSwKICAgICAgICAgICAgInNpemVfdmFyIiA9IHZhcihzaXplKSkgJT4lICAKICB1bmdyb3VwKCkgJT4lIAogIGNvbXBsZXRlKHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSkgJT4lIAogIGFycmFuZ2UoZGVzYyhzYW1wbGVfc2l6ZSkpCgphZHVsdF9zdW1tYXJpZXMgPSBmdWxsX2RhdGEgJT4lICAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2N0bWF4IiA9IG1lYW4oY3RtYXgpLAogICAgICAgICAgICAic2FtcGxlX3NpemUiID0gbigpLAogICAgICAgICAgICAiY3RtYXhfc3RfZXJyIiA9IChzZChjdG1heCkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJjdG1heF92YXIiID0gdmFyKGN0bWF4KSwgCiAgICAgICAgICAgICJtZWFuX3NpemUiID0gbWVhbihzaXplKSwKICAgICAgICAgICAgInNpemVfc3RfZXJyIiA9IChzZChzaXplKSAvIHNxcnQoc2FtcGxlX3NpemUpKSwKICAgICAgICAgICAgInNpemVfdmFyIiA9IHZhcihzaXplKSkgJT4lICAKICB1bmdyb3VwKCkgJT4lIAogIGNvbXBsZXRlKHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSkgJT4lIAogIGFycmFuZ2UoZGVzYyhzYW1wbGVfc2l6ZSkpCgoKdHNlcmllc19kYXRhID0gZnVsbF9kYXRhICU+JSAKICBtdXRhdGUoc3BfbmFtZSA9IGZjdF9yZW9yZGVyKHNwX25hbWUsIGN0bWF4LCAuZGVzYyA9IFQpKQoKZ2dwbG90KCkgKyAKICAjIGdlb21fdmxpbmUoZGF0YSA9IHVuaXF1ZShzZWxlY3QodHNlcmllc19kYXRhLCBjb2xsZWN0aW9uX2RhdGUpKSwgCiAgIyAgICAgICAgICAgIGFlcyh4aW50ZXJjZXB0ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpKSwKICAjICAgICAgICAgICAgY29sb3VyID0gImdyZXk5MCIsCiAgIyAgICAgICAgICAgIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9saW5lKGRhdGEgPSBjb2xsZWN0aW9uX2NvbmRpdGlvbnMsIAogICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoZGF0ZSksIHkgPSBtZWFuX3RlbXApLAogICAgICAgICAgICBjb2xvdXIgPSAiYmxhY2siLCAKICAgICAgICAgICAgbGluZXdpZHRoID0gMSkgKyAKICAjIGdlb21fZXJyb3JiYXIoZGF0YSA9IHNwZWNpZXNfc3VtbWFyaWVzLAogICMgICAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwKICAjICAgICAgICAgICAgICAgICAgIHltaW4gPSBtZWFuX2N0bWF4IC0gY3RtYXhfc3RfZXJyLCB5bWF4ID0gbWVhbl9jdG1heCArIGN0bWF4X3N0X2VyciwKICAjICAgICAgICAgICAgICAgICAgIGNvbG91ciA9IHNwX25hbWUpLAogICMgICAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMSksCiAgIyAgICAgICAgICAgICAgIHdpZHRoID0gNSwgbGluZXdpZHRoID0gMSkgKwogICMgZ2VvbV9wb2ludChkYXRhID0gYWR1bHRfc3VtbWFyaWVzLCAKICAjICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBtZWFuX2N0bWF4LCBjb2xvdXIgPSBzcF9uYW1lLCBzaXplID0gc2FtcGxlX3NpemUpKSArIAogIGdlb21fcG9pbnQoZGF0YSA9IHRzZXJpZXNfZGF0YSwgCiAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSwKICAgICAgICAgICAgIHNpemUgPSAxLjUsIHNoYXBlID0gMTYsIGFscGhhID0gMC45LAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIod2lkdGggPSAwLCBoZWlnaHQgPSAwKSkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBndWlkZXMoY29sb3VyID0gZ3VpZGVfbGVnZW5kKG92ZXJyaWRlLmFlcyA9IGxpc3QoYWxwaGEgPSAxLCBzaXplID0gNSkpKSArIAogIGxhYnMoeCA9ICJEYXRlIiwgCiAgICAgICB5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIsCiAgICAgICBzaXplID0gIlNhbXBsZSBTaXplIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiLCAKICAgICAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMyMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSkpCgpsYWtlX3RlbXBzID0gZ2dwbG90KCkgKyAKICBnZW9tX2xpbmUoZGF0YSA9IGNvbGxlY3Rpb25fY29uZGl0aW9ucywgCiAgICAgICAgICAgIGFlcyh4ID0gYXMuRGF0ZShkYXRlKSwgeSA9IG1lYW5fdGVtcCksCiAgICAgICAgICAgIGNvbG91ciA9ICJibGFjayIsIAogICAgICAgICAgICBsaW5ld2lkdGggPSAxKSArIAogIGxhYnMoeCA9ICJEYXRlIiwgCiAgICAgICB5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIsCiAgICAgICBzaXplID0gIlNhbXBsZSBTaXplIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQpgYGAKClRlbXBlcmF0dXJlcyBvYnNlcnZlZCBhdCB0aGUgdGltZSBvZiBjb2xsZWN0aW9uIGNsb3NlbHkgcmVzZW1ibGVkIHRoZSBtYXhpbXVtIGRhaWx5IHRlbXBlcmF0dXJlIGZyb20gdGhlIHRlbXBlcmF0dXJlIHNlbnNvciBkYXRhLiBNYXhpbXVtIHRlbXBlcmF0dXJlIHdhcyB1c2VkIGFzIGEgcHJveHkgaW5zdGVhZCBvZiBtZWFuIHRlbXBlcmF0dXJlIGFzIGNvbGxlY3Rpb25zIHdlcmUgdXN1YWxseSBtYWRlIGR1cmluZyBhZnRlcm5vb25zIG9yIGVhcmx5IGV2ZW5pbmdzLCBqdXN0IGZvbGxvd2luZyB0aGUgd2FybWVzdCBwYXJ0IG9mIHRoZSBkYXkuIAoKYGBge3Igc3VwcC1maWctdGVtcC1hY2MsIGZpZy53aWR0aCA9IDYsIGZpZy5oZWlnaHQ9Nn0KY29sbGVjdGlvbl9jb25kaXRpb25zICU+JSAKICBkcm9wX25hKGNvbGxlY3Rpb25fdGVtcCkgJT4lICAKICBnZ3Bsb3QoYWVzKHggPSBtYXhfdGVtcCwgeSA9IGNvbGxlY3Rpb25fdGVtcCkpICsgCiAgZ2VvbV9hYmxpbmUoaW50ZXJjZXB0ID0gMCwgc2xvcGUgPSAxLAogICAgICAgICAgICAgIGxpbmV3aWR0aCA9IDEsIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAzKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IGMoNSwxNSwyNSkpICsgCiAgc2NhbGVfeV9jb250aW51b3VzKGJyZWFrcyA9IGMoNSwxNSwyNSkpICsgCiAgbGFicyh4ID0gIk1heC4gVGVtcC4gZnJvbSBTZW5zb3IgKMKwQykiLAogICAgICAgeSA9ICJDb2xsZWN0aW9uIFRlbXAuICjCsEMpIikgKyAKICB0aGVtZV9tYXR0KCkKYGBgCgpgYGB7ciBtaXNjLXJvdW5kLXN1bW1hcnktMiwgZmlnLndpZHRoPTExLCBmaWcuaGVpZ2h0PTExLCBpbmNsdWRlID0gRn0KZ2dwbG90KCkgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAKICAgICAgICAgICAgICAgYyhtYXgoZnVsbF9kYXRhJGNvbGxlY3Rpb25fdGVtcCksCiAgICAgICAgICAgICAgICAgbWluKGZ1bGxfZGF0YSRjb2xsZWN0aW9uX3RlbXApKSwgCiAgICAgICAgICAgICBjb2xvdXIgPSAiZ3JleTYwIiwKICAgICAgICAgICAgIGxpbmV3aWR0aCA9IGMoMiwxKSwgCiAgICAgICAgICAgICBhbHBoYSA9IDAuNSkgKyAKICBnZW9tX2JhcihkYXRhID0gdW5pcXVlKHNlbGVjdChmdWxsX2RhdGEsIGNvbGxlY3Rpb25fZGF0ZSwgY29sbGVjdGlvbl90ZW1wKSksIAogICAgICAgICAgIGFlcyh4ID0gYXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gY29sbGVjdGlvbl90ZW1wKSwKICAgICAgICAgICBzdGF0ID0gImlkZW50aXR5IiwKICAgICAgICAgICBmaWxsID0gImdyZXkzMCIpICsgCiAgZ2VvbV9wb2ludChkYXRhID0gZnVsbF9kYXRhLCAKICAgICAgICAgICAgIGFlcyh4ID0gYXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gY3RtYXgpLAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIod2lkdGggPSAwLjcsIGhlaWdodCA9IDApLAogICAgICAgICAgICAgY29sb3VyID0gImdyZXkzMCIsCiAgICAgICAgICAgICBhbHBoYSA9IDAuNSkgKyAKICB5bGltKC0zLCA0MCkgKyAKICBjb29yZF9wb2xhcihzdGFydCA9IDApICsgCiAgdGhlbWVfdm9pZCgpCmBgYAoKClNpemUgYWxzbyB2YXJpZWQsIGJ1dCBwcmltYXJpbHkgYmV0d2VlbiByYXRoZXIgdGhhbiB3aXRoaW4gc3BlY2llcy4gCgpgYGB7ciBzdXBwLWZpZy1zaXplLXRpbWVzZXJpZXMsIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD01fQpnZ3Bsb3QoKSArIAogIGdlb21fdmxpbmUoZGF0YSA9IHVuaXF1ZShzZWxlY3QoZnVsbF9kYXRhLCBjb2xsZWN0aW9uX2RhdGUpKSwgCiAgICAgICAgICAgICBhZXMoeGludGVyY2VwdCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSksCiAgICAgICAgICAgICBjb2xvdXIgPSAiZ3JleTkwIiwKICAgICAgICAgICAgIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9saW5lKGRhdGEgPSBjb2xsZWN0aW9uX2NvbmRpdGlvbnMsIAogICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoZGF0ZSksIHkgPSBtZWFuX3RlbXApLAogICAgICAgICAgICBjb2xvdXIgPSAiYmxhY2siLCAKICAgICAgICAgICAgbGluZXdpZHRoID0gMikgKyAKICAjIGdlb21fZXJyb3JiYXIoZGF0YSA9IHNwZWNpZXNfc3VtbWFyaWVzLAogICMgICAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgCiAgIyAgICAgICAgICAgICAgICAgICB5bWluID0gbWVhbl9jdG1heCAtIGN0bWF4X3N0X2VyciwgeW1heCA9IG1lYW5fY3RtYXggKyBjdG1heF9zdF9lcnIsCiAgIyAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBzcF9uYW1lKSwKICAjICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDEpLAogICMgICAgICAgICAgICAgICB3aWR0aCA9IDUsIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9wb2ludChkYXRhID0gYWR1bHRfc3VtbWFyaWVzLCAKICAgICAgICAgICAgIGFlcyh4ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gbWVhbl9zaXplICogNDAsIGNvbG91ciA9IHNwX25hbWUsIHNpemUgPSBzYW1wbGVfc2l6ZSksCiAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMSkpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgc2NhbGVfeV9jb250aW51b3VzKAogICAgbmFtZSA9ICJUZW1wZXJhdHVyZSIsICMgRmVhdHVyZXMgb2YgdGhlIGZpcnN0IGF4aXMKICAgIHNlYy5heGlzID0gc2VjX2F4aXMofi4vNDAsIG5hbWU9IlByb3NvbWUgTGVuZ3RoIChtbSkiKSwgIyBBZGQgYSBzZWNvbmQgYXhpcyBhbmQgc3BlY2lmeSBpdHMgZmVhdHVyZXMKICAgIGJyZWFrcyA9IGMoMCw1LDEwLDE1LDIwLDI1LDMwKQogICkgKyAKICBsYWJzKHggPSAiRGF0ZSIsIAogICAgICAgeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsIAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKYGBge3IgbWlzYy10cmFpdC1kb3ktZmVhdHVyZSwgZmlnLndpZHRoID0gMTQsIGZpZy5oZWlnaHQgPSA3LCBpbmNsdWRlID0gRn0KI1Nob3duIGJlbG93IGlzIENUbWF4IGFuZCBib2R5IHNpemUgZm9yIHRoZSBzcGVjaWVzIHdpdGggdGhlIG1vc3QgZGF0YSAoKlNraXN0b2RpYXB0b211cyosICpMLiBtaW51dHVzKiwgKkwuIHNpY2lsaXMqLCBhbmQgKkVwaXNjaHVyYSopLCBwbG90dGVkIGFnYWluc3QgdGhlIGRheSBvZiB0aGUgeWVhciBmb3IgZWFjaCBzZXgvc3RhZ2Ugc2VwYXJhdGVseS4gCgpjdG1heF9mZWF0dXJlID0gZnVsbF9kYXRhICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgYygiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIiwgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiLCAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIsICJFcGlzY2h1cmEgbGFjdXN0cmlzIikpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBjdG1heCwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfZ3JpZChzcF9uYW1lfnNleCkgKyAKICBnZW9tX3BvaW50KCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiRGF5IG9mIHRoZSBZZWFyIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIikgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMwMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSksCiAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQoKc2l6ZV9mZWF0dXJlID0gZnVsbF9kYXRhICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgYygiU2tpc3RvZGlhcHRvbXVzIG9yZWdvbmVuc2lzIiwgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiLCAiTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyIsICJFcGlzY2h1cmEgbGFjdXN0cmlzIikpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBzaXplLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF9ncmlkKHNwX25hbWV+c2V4KSArIAogIGdlb21fcG9pbnQoKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgdGhlIFllYXIiLCAKICAgICAgIHkgPSAiU2l6ZSAobW0pIikgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsKICB0aGVtZShheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDMwMCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSksCiAgICAgICAgbGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQoKZ2dhcnJhbmdlKGN0bWF4X2ZlYXR1cmUsIHNpemVfZmVhdHVyZSwgY29tbW9uLmxlZ2VuZCA9IFQsIGxhYmVscyA9ICJBVVRPIiwgbGVnZW5kID0gIm5vbmUiKQpgYGAKCmBgYHtyIHNwLW9jY3VyZW5jZXMsIGZpZy53aWR0aD03LCBmaWcuaGVpZ2h0PTV9CnNhbXBsZV9kYXRlc19wbG90ID0gZnVsbF9kYXRhICU+JSAgCiAgZmlsdGVyKHNwX25hbWUgIT0gIk9zcGhyYW50aWN1bSBsYWJyb25lY3R1bSIpICU+JSAKICBtdXRhdGUoc3BfbmFtZSA9IGFzLmZhY3RvcihzcF9uYW1lKSwKICAgICAgICAgc3BfbmFtZSA9IGZjdF9yZW9yZGVyKHNwX25hbWUsIGN0bWF4KSkgJT4lIAogIGdncGxvdChhZXMoeCA9IGx1YnJpZGF0ZTo6YXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpLCAKICAgICAgICAgICAgIHkgPSBzcF9uYW1lLCBmaWxsID0gc3BfbmFtZSkpICsgCiAgIyBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSBhc19kYXRlKAogICMgICBjKCIyMDIzLTA1LTAxIiwKICAjICAgICAiMjAyMy0wOS0wMSIsCiAgIyAgICAgIjIwMjQtMDEtMDEiLAogICMgICAgICIyMDI0LTA1LTAxIikpLAogICMgICBjb2xvdXIgPSAiZ3JleSIsCiAgIyAgIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9kZW5zaXR5X3JpZGdlcyhiYW5kd2lkdGggPSAzMCwKICAgICAgICAgICAgICAgICAgICAgIGppdHRlcmVkX3BvaW50cyA9IFRSVUUsIAogICAgICAgICAgICAgICAgICAgICAgcG9pbnRfc2hhcGUgPSAyMSwKICAgICAgICAgICAgICAgICAgICAgIHBvaW50X3NpemUgPSAxLAogICAgICAgICAgICAgICAgICAgICAgcG9pbnRfY29sb3VyID0gImdyZXkzMCIsCiAgICAgICAgICAgICAgICAgICAgICBwb2ludF9hbHBoYSA9IDAuOCwKICAgICAgICAgICAgICAgICAgICAgIGFscGhhID0gMC44LAogICAgICAgICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9wb2ludHNfaml0dGVyKAogICAgICAgICAgICAgICAgICAgICAgICBoZWlnaHQgPSAwLjEsIHdpZHRoID0gMCkpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHNjYWxlX3hfZGF0ZShkYXRlX2JyZWFrcyA9ICIzIG1vbnRocyIsCiAgICAgICAgICAgICAgIGRhdGVfbGFiZWxzID0gIiViIikgKyAKICBjb29yZF9jYXJ0ZXNpYW4oeGxpbSA9IGx1YnJpZGF0ZTo6YXNfZGF0ZShjKCIyMDIzLTA2LTA4IiwgIjIwMjQtMDUtMDgiKSkpICsgCiAgbGFicyh4ID0gIkRheSBvZiBZZWFyIiwgCiAgICAgICB5ID0gIlNwZWNpZXMiKSArIAogIHRoZW1lX21hdHQoKSArIAogICN0aGVtZV9yaWRnZXMoZ3JpZCA9IFQpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKYGBgCgpUaGUgc2FtcGxlcyBjYXB0dXJlZCB0aGUgYnJvYWQgc2Vhc29uYWwgY2hhbmdlcyBpbiBjYWxhbm9pZCBjb3BlcG9kIGNvbW11bml0eSBjb21wb3NpdGlvbiBpbiB0aGUgbGFrZS4gV2Ugbm90ZSwgaG93ZXZlciwgdGhhdCByYXJlIHNwZWNpZXMgKGUuZy4gKlNlbmVjZWxsYSogYW5kICpMaW1ub2NhbGFudXMqKSB3ZXJlIG9mdGVuIHByZWZlcmVudGlhbGx5IHNhbXBsZWQsIHNvIGFyZSBvdmVyLXJlcHJlc2VudGVkIGluIHRoZSBkYXRhIHNldC4gCgpgYGB7ciBzdXBwLWZpZy1zcC1wcm9wcywgZmlnLndpZHRoID0gMTIsIGZpZy5oZWlnaHQgPSA1LCBpbmNsdWRlID0gRn0KYWR1bHRfc3VtbWFyaWVzICU+JSAKICB1bmdyb3VwKCkgJT4lIAogIG11dGF0ZShjb2xsZWN0aW9uX251bSA9IGFzLm51bWVyaWMoZmFjdG9yKGNvbGxlY3Rpb25fZGF0ZSkpKSAlPiUgCiAgZ3JvdXBfYnkoY29sbGVjdGlvbl9kYXRlKSAlPiUgIAogIGFycmFuZ2UoY29sbGVjdGlvbl9kYXRlKSAlPiUgCiAgc2VsZWN0KHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSwgY29sbGVjdGlvbl9udW0sIHNhbXBsZV9zaXplKSAlPiUgCiAgbXV0YXRlKHNhbXBsZV9zaXplID0gcmVwbGFjZV9uYShzYW1wbGVfc2l6ZSwgMCkpICU+JSAKICBtdXRhdGUodG90YWwgPSBzdW0oc2FtcGxlX3NpemUpLAogICAgICAgICBwZXJjZW50YWdlID0gc2FtcGxlX3NpemUgLyB0b3RhbCwKICAgICAgICAgY29sbGVjdGlvbl9kYXRlID0gbHVicmlkYXRlOjphc19kYXRlKGNvbGxlY3Rpb25fZGF0ZSkpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBjb2xsZWN0aW9uX2RhdGUsIHkgPSBwZXJjZW50YWdlLCBmaWxsID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9hcmVhKCkgKyAKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgc2NhbGVfeV9jb250aW51b3VzKGJyZWFrcyA9IGMoMCwxKSkgKyAKICBsYWJzKHggPSAiQ29sbGVjdGlvbiBEYXRlIiwgCiAgICAgICB5ID0gIlByb3BvcnRpb24iLCAKICAgICAgIGZpbGwgPSAiU3BlY2llcyIpICsgCiAgdGhlbWVfbWluaW1hbChiYXNlX3NpemUgPSAyMCkgKyAKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGlja3MgPSBlbGVtZW50X2xpbmUoKSkKYGBgCgpUaHJvdWdob3V0IHRoZSBzZWFzb24sIHRoZSBwcmV2YWxlbmNlIG9mIHZhcmlvdXMgdW5pZGVudGlmaWVkIHBhdGhvZ2VucyBhbHNvIHZhcmllZCwgd2l0aCB2ZXJ5IGxpdHRsZSBpbmZlY3Rpb24gb2JzZXJ2ZWQgZHVyaW5nIHRoZSBXaW50ZXIgYW5kIFNwcmluZy4gCgpgYGB7ciBzdXBwLWZpZy1wYXRob2dlbi1wcm9wcywgZmlnLndpZHRoPTgsIGZpZy5oZWlnaHQ9NX0KcGF0aG9nZW5fY29scyA9IGMoIm5vIiA9ICJncmV5OTUiLCAiY2xvdWR5IiA9ICJob25leWRldzMiLCAic3BvdCIgPSAiYW50aXF1ZXdoaXRlMyIsICJvdGhlciIgPSAidG9tYXRvMyIpCgpmdWxsX2RhdGEgJT4lIAogIHNlbGVjdChjb2xsZWN0aW9uX2RhdGUsIGRldl9lZ2dzLCBwYXRob2dlbiwgbGlwaWRzLCBzcF9uYW1lLCBzZXgpICU+JSAKICBncm91cF9ieSgpICU+JSAKICBmaWx0ZXIoc2V4ICE9ICJqdXZlbmlsZSIpICU+JSAKICBncm91cF9ieShjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBjb3VudChwYXRob2dlbikgJT4lIAogIGZpbHRlcihwYXRob2dlbiAhPSAidW5jZXJ0YWluIikgJT4lIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSAiY29sbGVjdGlvbl9kYXRlIiwgCiAgICAgICAgICAgICAgbmFtZXNfZnJvbSA9IHBhdGhvZ2VuLCAKICAgICAgICAgICAgICB2YWx1ZXNfZnJvbSA9IG4sCiAgICAgICAgICAgICAgdmFsdWVzX2ZpbGwgPSAwKSAlPiUgCiAgbXV0YXRlKHRvdGFsID0gc3VtKG5vLCBjbG91ZHksIHNwb3QsIG90aGVyKSkgJT4lIAogIHBpdm90X2xvbmdlcihjb2xzID0gYyhubywgY2xvdWR5LCBzcG90LCBvdGhlciksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhdGhvZ2VuIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJjb3VudCIpICU+JSAKICBtdXRhdGUocGVyY2VudCA9IGNvdW50L3RvdGFsLAogICAgICAgICBjb2xsZWN0aW9uX2RhdGUgPSBsdWJyaWRhdGU6OmFzX2RhdGUoY29sbGVjdGlvbl9kYXRlKSwKICAgICAgICAgcGF0aG9nZW4gPSBmY3RfcmVsZXZlbChwYXRob2dlbiwgIm5vIiwgImNsb3VkeSIsICJzcG90IiwgIm90aGVyIikpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBjb2xsZWN0aW9uX2RhdGUsIHkgPSBwZXJjZW50LCBmaWxsID0gcGF0aG9nZW4pKSArIAogIGdlb21fYXJlYSgpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gcGF0aG9nZW5fY29scykgKyAKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzID0gYygwLDEpKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIERhdGUiLCAKICAgICAgIHkgPSAiUHJvcG9ydGlvbiIsIAogICAgICAgZmlsbCA9ICJQYXRob2dlbiIpICsgCiAgdGhlbWVfbWluaW1hbChiYXNlX3NpemUgPSAyMCkgKyAKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGlja3MgPSBlbGVtZW50X2xpbmUoKSkKYGBgCgpUaGUgdHJhbnNwYXJlbnQgYm9kaWVzIG9mIHRoZXNlIGNvcGVwb2RzIGFsc28gYWxsb3dlZCB1cyB0byBleGFtaW5lIHNlYXNvbmFsIHBhdHRlcm5zIGluIGxpcGlkIHJlc2VydmVzIGFuZCBpbiB0aGUgcHJvZHVjdGlvbiBvZiBlZ2dzLiBNYXR1cmluZyBvb2N5dGVzIGFyZSB2aXNpYmxlIGluIGZlbWFsZSBjb3BlcG9kcyBiZWZvcmUgdGhleSBhcmUgcmVsZWFzZWQuIFRoZXJlIHdhcyBubyBzdHJvbmcgc2Vhc29uYWwgY3ljbGUgaW4gdGhlIHByb2R1Y3Rpb24gb2YgdGhlc2UgZWdncyBpbiBhbnkgc3BlY2llcywgYW5kIGluc3RlYWQsIGZlbWFsZXMgd2VyZSByZXByb2R1Y3RpdmVseSBhY3RpdmUgdGhyb3VnaG91dCB0aGVpciByZXNwZWN0aXZlIHNlYXNvbnMgb2Ygb2NjdXJlbmNlLiAKCmBgYHtyIHN1cHAtZmlnLWRldmVnZ3MtcHJvcHMsIGZpZy5oZWlnaHQgPSAxNSwgZmlnLndpZHRoID0gOH0KZGV2X2VnZ3NfY29scyA9IGMoIm5vIiA9ICJncmV5OTUiLCAieWVzIiA9ICJsaWdodGJsdWUzIikKCmZ1bGxfZGF0YSAlPiUgCiAgc2VsZWN0KGNvbGxlY3Rpb25fZGF0ZSwgZGV2X2VnZ3MsIHBhdGhvZ2VuLCBsaXBpZHMsIHNwX25hbWUsIHNleCkgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JSAKICBmaWx0ZXIoc2V4ICE9ICJqdXZlbmlsZSIpICU+JSAKICBncm91cF9ieShzcF9uYW1lLCBjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBjb3VudChkZXZfZWdncykgJT4lIAogIGZpbHRlcihkZXZfZWdncyAhPSAidW5jZXJ0YWluIikgJT4lIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSBjKCJjb2xsZWN0aW9uX2RhdGUiLCAic3BfbmFtZSIpLCAKICAgICAgICAgICAgICBuYW1lc19mcm9tID0gZGV2X2VnZ3MsIAogICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbiwKICAgICAgICAgICAgICB2YWx1ZXNfZmlsbCA9IDApICU+JSAKICBtdXRhdGUodG90YWwgPSBzdW0obm8sIHllcykpICU+JSAKICBwaXZvdF9sb25nZXIoY29scyA9IGMobm8sIHllcyksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gImRldl9lZ2dzIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJjb3VudCIpICU+JSAKICBtdXRhdGUocGVyY2VudCA9IGNvdW50L3RvdGFsLAogICAgICAgICBjb2xsZWN0aW9uX2RhdGUgPSBsdWJyaWRhdGU6OmFzX2RhdGUoY29sbGVjdGlvbl9kYXRlKSwKICAgICAgICAgZGV2X2VnZ3MgPSBmY3RfcmVsZXZlbChkZXZfZWdncywgIm5vIiwgInllcyIpKSAlPiUgCiAgdW5ncm91cCgpICU+JSAKICBjb21wbGV0ZShjb2xsZWN0aW9uX2RhdGUsIG5lc3Rpbmcoc3BfbmFtZSwgZGV2X2VnZ3MpLCBmaWxsID0gbGlzdChwZXJjZW50ID0gMSkpICU+JSAKICBtdXRhdGUocGVyY2VudCA9IGlmX2Vsc2UoaXMubmEodG90YWwpICYgZGV2X2VnZ3MgPT0gInllcyIsIDAsIHBlcmNlbnQpKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gY29sbGVjdGlvbl9kYXRlLCB5ID0gcGVyY2VudCwgZmlsbCA9IGRldl9lZ2dzKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+LiwgbmNvbCA9IDEpICsgCiAgZ2VvbV9hcmVhKCkgKyAKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBkZXZfZWdnc19jb2xzKSArIAogIHNjYWxlX3lfY29udGludW91cyhicmVha3MgPSBjKDAsMSkpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gRGF0ZSIsIAogICAgICAgeSA9ICJQcm9wb3J0aW9uIiwgCiAgICAgICBmaWxsID0gIkRldmVsb3BpbmcgXG5FZ2dzIikgKyAKICB0aGVtZV9taW5pbWFsKGJhc2Vfc2l6ZSA9IDIwKSArIAogIHRoZW1lKHBhbmVsLmdyaWQgPSBlbGVtZW50X2JsYW5rKCksCiAgICAgICAgYXhpcy50aWNrcyA9IGVsZW1lbnRfbGluZSgpKQpgYGAKClRoZSBwcmVzZW5jZSBvZiBsaXBpZHMgdmFyaWVkIGFjcm9zcyBzcGVjaWVzLCB3aXRoIG9ubHkgKkwuIG1pbnV0dXMqLCAqTC4gc2ljaWxpcyosIGFuZCAqTGltbm9jYWxhbnVzKiByZWd1bGFybHkgcG9zc2Vzc2luZyBsaXBpZCBzdG9yZXMuIAoKYGBge3Igc3VwcC1maWctbGlwaWRzLXByb3BzLCBmaWcuaGVpZ2h0ID0gMTUsIGZpZy53aWR0aCA9IDh9CmxpcGlkX2NvbHMgPSBjKCJubyIgPSAiZ3JleTk1IiwgInllcyIgPSAic2llbm5hMiIpCgpmdWxsX2RhdGEgJT4lIAogIHNlbGVjdChjb2xsZWN0aW9uX2RhdGUsIGRldl9lZ2dzLCBwYXRob2dlbiwgbGlwaWRzLCBzcF9uYW1lLCBzZXgpICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKHNleCAhPSAianV2ZW5pbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlKSAlPiUgCiAgY291bnQobGlwaWRzKSAlPiUgCiAgZmlsdGVyKGxpcGlkcyAhPSAidW5jZXJ0YWluIikgJT4lIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSBjKCJjb2xsZWN0aW9uX2RhdGUiLCAic3BfbmFtZSIpLCAKICAgICAgICAgICAgICBuYW1lc19mcm9tID0gbGlwaWRzLCAKICAgICAgICAgICAgICB2YWx1ZXNfZnJvbSA9IG4sCiAgICAgICAgICAgICAgdmFsdWVzX2ZpbGwgPSAwKSAlPiUgCiAgbXV0YXRlKHRvdGFsID0gc3VtKG5vLCB5ZXMpKSAlPiUgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKG5vLCB5ZXMpLAogICAgICAgICAgICAgICBuYW1lc190byA9ICJsaXBpZHMiLCAKICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gImNvdW50IikgJT4lIAogIG11dGF0ZShwZXJjZW50ID0gY291bnQvdG90YWwsCiAgICAgICAgIGNvbGxlY3Rpb25fZGF0ZSA9IGx1YnJpZGF0ZTo6YXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpLAogICAgICAgICBsaXBpZHMgPSBmY3RfcmVsZXZlbChsaXBpZHMsICJubyIsICJ5ZXMiKSkgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgY29tcGxldGUoY29sbGVjdGlvbl9kYXRlLCBuZXN0aW5nKHNwX25hbWUsIGxpcGlkcyksIGZpbGwgPSBsaXN0KHBlcmNlbnQgPSAxKSkgJT4lIAogIG11dGF0ZShwZXJjZW50ID0gaWZfZWxzZShpcy5uYSh0b3RhbCkgJiBsaXBpZHMgPT0gInllcyIsIDAsIHBlcmNlbnQpKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gY29sbGVjdGlvbl9kYXRlLCB5ID0gcGVyY2VudCwgZmlsbCA9IGxpcGlkcykpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4sIG5jb2wgPSAxKSArIAogIGdlb21fYXJlYSgpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gbGlwaWRfY29scykgKyAKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzID0gYygwLDEpKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIERhdGUiLCAKICAgICAgIHkgPSAiUHJvcG9ydGlvbiIsIAogICAgICAgZmlsbCA9ICJMaXBpZHNcblByZXNlbnQiKSArIAogIHRoZW1lX21pbmltYWwoYmFzZV9zaXplID0gMjApICsgCiAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKICAgICAgICBheGlzLnRpY2tzID0gZWxlbWVudF9saW5lKCkpCmBgYAoKCiMjIFRlbXBlcmF0dXJlIFZhcmlhYmlsaXR5Ckxha2UgQ2hhbXBsYWluIGlzIGhpZ2hseSBzZWFzb25hbCwgd2l0aCBib3RoIGF2ZXJhZ2UgdGVtcGVyYXR1cmVzIGFuZCB0ZW1wZXJhdHVyZSB2YXJpYWJpbGl0eSBjaGFuZ2luZyB0aHJvdWdob3V0IHRoZSB5ZWFyLiBUaGVzZSBwYXR0ZXJucyBpbiB0aGUgZXhwZXJpZW5jZWQgdGhlcm1hbCBlbnZpcm9ubWVudCBtYXkgZHJpdmUgdGhlIG9ic2VydmVkIHZhcmlhdGlvbiBpbiBjb3BlcG9kIHRoZXJtYWwgbGltaXRzLiBIb3dldmVyLCB0aGUgdGltZSBwZXJpb2QgYWZmZWN0aW5nIGNvcGVwb2QgdGhlcm1hbCBsaW1pdHMgaXMgdW5rbm93bi4gRGVwZW5kaW5nIHRoZSBvbiB0aGUgZHVyYXRpb24gb2YgdGltZSBjb25zaWRlcmVkLCB0aGVyZSBhcmUgbGFyZ2UgY2hhbmdlcyBpbiB0aGUgZXhwZXJpZW5jZWQgZW52aXJvbm1lbnQsIGluIHBhcnRpY3VsYXIgcmVnYXJkaW5nIHRoZSB0ZW1wZXJhdHVyZSByYW5nZSBhbmQgdmFyaWFuY2UuIENvbnNpZGVyIGZvciBleGFtcGxlIHRocmVlIHRpbWUgcGVyaW9kczogdGhlIGRheSBvZiBjb2xsZWN0aW9uLCBvbmUgd2VlayBwcmlvciB0byBjb2xsZWN0aW9uLCBhbmQgZm91ciB3ZWVrcyBwcmlvciB0byBjb2xsZWN0aW9uLiBXaGlsZSB0aGUgb3ZlcmFsbCBwYXR0ZXJuIGlzIHNpbWlsYXIsIHdlIGNhbiBzZWUgdGhhdCwgdW5zdXJwcmlzaW5nbHksIGNvbnNpZGVyaW5nIGxvbmdlciBwZXJpb2RzIG9mIHRpbWUgcmVzdWx0cyBpbiBsYXJnZXIgcmFuZ2VzIGFuZCBzbGlnaHRseSBjaGFuZ2VzIHRoZSBwYXR0ZXJuIG9mIHZhcmlhbmNlIGV4cGVyaWVuY2VkLiAKCmBgYHtyIGRhaWx5LXRlbXAtZGF0YSwgaW5jbHVkZSA9IEZ9CiMjIERhaWx5IHZhbHVlcwpkYWlseV90ZW1wX2RhdGEgPSB0ZW1wX2RhdGEgJT4lCiAgdW5ncm91cCgpICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXNlKG1lYW5fdGVtcCA9IG1lYW4odGVtcCksCiAgICAgICAgICAgIG1lZF90ZW1wID0gbWVkaWFuKHRlbXApLAogICAgICAgICAgICB2YXJfdGVtcCA9IHZhcih0ZW1wKSwgCiAgICAgICAgICAgIG1pbl90ZW1wID0gbWluKHRlbXApLCAKICAgICAgICAgICAgbWF4X3RlbXAgPSBtYXgodGVtcCkpICU+JSAKICBtdXRhdGUoInJhbmdlX3RlbXAiID0gbWF4X3RlbXAgLSBtaW5fdGVtcCkKCmRheV9wcmlvcl90ZW1wX2RhdGEgPSB0ZW1wX2RhdGEgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcmlzZShtZWFuX3RlbXAgPSBtZWFuKHRlbXApLAogICAgICAgICAgICBtZWRfdGVtcCA9IG1lZGlhbih0ZW1wKSwKICAgICAgICAgICAgdmFyX3RlbXAgPSB2YXIodGVtcCksIAogICAgICAgICAgICBtaW5fdGVtcCA9IG1pbih0ZW1wKSwgCiAgICAgICAgICAgIG1heF90ZW1wID0gbWF4KHRlbXApKSAlPiUgCiAgbXV0YXRlKGRhdGUgPSBkYXRlICsgMSkgJT4lIAogIHJlbmFtZV93aXRoKC5mbiA9IH4gcGFzdGUwKCJwcmlvcl9kYXlfIiwgLngpLCAuY29scyA9IGMoLWRhdGUpKQoKZGFpbHlfcGxvdCA9IGRhaWx5X3RlbXBfZGF0YSAlPiUgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lIAogIGdncGxvdChhZXMoeCA9IGRhdGUsIHkgPSB0ZW1wLCBjb2xvdXIgPSBwYXJhbWV0ZXIpKSArIAogIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKICAgICJtZWFuX3RlbXAiID0gIm9saXZlZHJhYjMiLAogICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAogICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLCAgCiAgICAibWluX3RlbXAiID0gImRvZGdlcmJsdWUiLAogICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAogICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKICApKSArIAogIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAogIGdndGl0bGUoIkRhaWx5IFZhbHVlcyIpICsgCiAgbGFicyh5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwKICAgICAgIHggPSAiIikgKyAKICB0aGVtZV9idyhiYXNlX3NpemUgPSAyMCkgKyAKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKYGBgCgpgYGB7ciBwcmVkaWN0b3JzLWZ1bmN0aW9ufQojIyBEZWZpbmluZyB0aGUgZnVuY3Rpb24gdG8gZ2V0IHByZWRpY3RvciB2YWx1ZXMgZm9yIHBlcmlvZHMgb2YgZGlmZmVyZW50IGxlbmd0aHMKZ2V0X3ByZWRpY3RvcnMgPSBmdW5jdGlvbihkYWlseV92YWx1ZXMsIHJhd190ZW1wLCBuX2RheXMpewogIHByZWZpeCA9IHN0cl9yZXBsYWNlX2FsbCh4ZnVuOjpudW1iZXJzX3RvX3dvcmRzKG5fZGF5cyksIHBhdHRlcm4gPSAiICIsIHJlcGxhY2VtZW50ID0gIi0iKQogIAogIG1lYW5fdmFsdWVzID0gZGFpbHlfdmFsdWVzICU+JSAKICAgIHVuZ3JvdXAoKSAlPiUgCiAgICBtdXRhdGUobWVhbl9tYXggPSBzbGlkZV92ZWMoLnggPSBtYXhfdGVtcCwgLmYgPSBtZWFuLCAuYmVmb3JlID0gbl9kYXlzLCAuY29tcGxldGUgPSBUKSwKICAgICAgICAgICBtZWFuX21pbiA9IHNsaWRlX3ZlYygueCA9IG1pbl90ZW1wLCAuZiA9IG1lYW4sIC5iZWZvcmUgPSBuX2RheXMsIC5jb21wbGV0ZSA9IFQpLAogICAgICAgICAgIG1lYW5fcmFuZ2UgPSBzbGlkZV92ZWMoLnggPSByYW5nZV90ZW1wLCAuZiA9IG1lYW4sIC5iZWZvcmUgPSBuX2RheXMsIC5jb21wbGV0ZSA9IFQpKSAlPiUgCiAgICBzZWxlY3QoZGF0ZSwgbWVhbl9tYXgsIG1lYW5fbWluLCBtZWFuX3JhbmdlKSAlPiUgCiAgICByZW5hbWVfd2l0aCggfiBwYXN0ZShwcmVmaXgsICJkYXkiLCAueCwgc2VwID0gIl8iKSwgLmNvbHMgPSBjKC1kYXRlKSkKICAKICBwZXJpb2RfdmFsdWVzID0gcmF3X3RlbXAgJT4lIAogICAgbXV0YXRlKG1lYW4gPSBzbGlkZV9pbmRleF9tZWFuKHRlbXAsIGkgPSBkYXRlLCBiZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5hX3JtID0gVCksCiAgICAgICAgICAgbWF4ID0gc2xpZGVfaW5kZXhfbWF4KHRlbXAsIGkgPSBkYXRlLCBiZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQpLAogICAgICAgICAgIG1pbiA9IHNsaWRlX2luZGV4X21pbih0ZW1wLCBpID0gZGF0ZSwgYmVmb3JlID0gZGF5cyhuX2RheXMpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQpLAogICAgICAgICAgIG1lZCA9IHNsaWRlX2luZGV4X2RibCh0ZW1wLCAuaSA9IGRhdGUsIC5iZWZvcmUgPSBkYXlzKG5fZGF5cyksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQsIC5mID0gbWVkaWFuKSwKICAgICAgICAgICB2YXIgPSBzbGlkZV9pbmRleF9kYmwodGVtcCwgLmkgPSBkYXRlLCAuYmVmb3JlID0gZGF5cyhuX2RheXMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLmYgPSB2YXIpLAogICAgICAgICAgIHJhbmdlID0gbWF4IC0gbWluKSAlPiUgIAogICAgc2VsZWN0KC10ZW1wKSAlPiUgIAogICAgZGlzdGluY3QoKSAlPiUgCiAgICByZW5hbWVfd2l0aCggfiBwYXN0ZShwcmVmaXgsICJkYXkiLCAueCwgc2VwID0gIl8iKSwgLmNvbHMgPSBjKC1kYXRlKSklPiUgCiAgICBpbm5lcl9qb2luKG1lYW5fdmFsdWVzLCBieSA9IGMoImRhdGUiKSkgJT4lICAKICAgIGRyb3BfbmEoKQogIAogIHJldHVybihwZXJpb2RfdmFsdWVzKQp9CmBgYAoKYGBge3IgbWlzYy1wcmVkaWN0b3JzLWFuZC1wbG90cywgZmlnLndpZHRoPTEyLCBmaWcuaGVpZ2h0PTUsIGluY2x1ZGUgPSBGfQojICMjIEdldHRpbmcgcHJlZGljdG9yIHZhcmlhYmxlcyBmb3IgZGlmZmVyZW50IHBlcmlvZHMKIyAKIyAjIyMgU2hvcnQgKHRocmVlIGRheXMpCiMgdGhyZWVfZGF5X3RlbXBzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gZGFpbHlfdGVtcF9kYXRhLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHRlbXBfZGF0YSwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gMykKIyAKIyAjIyMgT05FIFdFRUsKd2Vla190ZW1wcyA9IGdldF9wcmVkaWN0b3JzKGRhaWx5X3ZhbHVlcyA9IGRhaWx5X3RlbXBfZGF0YSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gdGVtcF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gNykKCndlZWtfcGxvdCA9IHdlZWtfdGVtcHMgJT4lCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwKICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUKICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygic2V2ZW5fZGF5X21lYW4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfbWVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAic2V2ZW5fZGF5X21heCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInNldmVuX2RheV9taW4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfdmFyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAic2V2ZW5fZGF5X3JhbmdlIikpICU+JQogIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUKICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKwogIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAogICAgIm1lYW5fdGVtcCIgPSAib2xpdmVkcmFiMyIsCiAgICAibWVkX3RlbXAiID0gInNlYWdyZWVuMyIsCiAgICAibWF4X3RlbXAiID0gInRvbWF0byIsCiAgICAibWluX3RlbXAiID0gImRvZGdlcmJsdWUiLAogICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAogICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKICApKSArCiAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IGFzLkRhdGUoYygiMjAyMy0wMS0wMSIsICIyMDIzLTA0LTAxIiwgIjIwMjMtMDctMDEiKSkpICsKICBnZ3RpdGxlKCJPbmUgV2VlayIpICsKICBsYWJzKHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLAogICAgICAgeCA9ICIiKSArCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgVFdPIFdFRUtTCiMgdHdvX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gMTQpCiMgCiMgdHdvX3dlZWtfcGxvdCA9IHR3b193ZWVrX3RlbXBzICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUgCiMgICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygiZm91cnRlZW5fZGF5X21lYW4iLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tZWQiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tYXgiLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmb3VydGVlbl9kYXlfbWluIiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3ZhciIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3JhbmdlIikpICU+JSAKIyAgIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKIyAgIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAojICAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKIyAgICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAojICAgICAibWF4X3RlbXAiID0gInRvbWF0byIsICAKIyAgICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKIyAgICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAojICAgICAidmFyX3RlbXAiID0gImRhcmtnb2xkZW5yb2QxIgojICAgKSkgKyAKIyAgIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAojICAgZ2d0aXRsZSgiVHdvIFdlZWtzIikgKyAKIyAgIGxhYnMoeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsCiMgICAgICAgIHggPSAiIikgKyAKIyAgIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDIwKSArIAojICAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKIyAgICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgRk9VUiBXRUVLUwpmb3VyX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gdGVtcF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSAyOCkKCmZvdXJfd2Vla19wbG90ID0gZm91cl93ZWVrX3RlbXBzICU+JQogIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhcmFtZXRlciIsCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lCiAgZmlsdGVyKHBhcmFtZXRlciAlaW4lIGMoInR3ZW50eS1laWdodF9kYXlfbWVhbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInR3ZW50eS1laWdodF9kYXlfbWVkIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9tYXgiLAogICAgICAgICAgICAgICAgICAgICAgICAgICJ0d2VudHktZWlnaHRfZGF5X21pbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgInR3ZW50eS1laWdodF9kYXlfdmFyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9yYW5nZSIpKSAlPiUKICBtdXRhdGUocGFyYW1ldGVyID0gcGFzdGUod29yZChwYXJhbWV0ZXIsIHN0YXJ0ID0gMywgc2VwID0gZml4ZWQoIl8iKSksICJfdGVtcCIsIHNlcCA9ICIiKSkgJT4lCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXAsIGNvbG91ciA9IHBhcmFtZXRlcikpICsKICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKwogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKICAgICJtZWFuX3RlbXAiID0gIm9saXZlZHJhYjMiLAogICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAogICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLAogICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKICAgICJyYW5nZV90ZW1wIiA9ICJnb2xkZW5yb2QzIiwKICAgICJ2YXJfdGVtcCIgPSAiZGFya2dvbGRlbnJvZDEiCiAgKSkgKwogIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArCiAgZ2d0aXRsZSgiRm91ciBXZWVrcyIpICsKICBsYWJzKHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLAogICAgICAgeCA9ICIiKSArCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsKICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgRUlHSFQgV0VFS1MKIyBlaWdodF93ZWVrX3RlbXBzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gZGFpbHlfdGVtcF9kYXRhLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSA1NikKIyAKIyBlaWdodF93ZWVrX3Bsb3QgPSBlaWdodF93ZWVrX3RlbXBzICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUgCiMgICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygiZmlmdHktc2l4X2RheV9tZWFuIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X21lZCIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZmlmdHktc2l4X2RheV9tYXgiLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X21pbiIsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZpZnR5LXNpeF9kYXlfdmFyIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X3JhbmdlIikpICU+JSAKIyAgIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKIyAgIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAojICAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKIyAgICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAojICAgICAibWF4X3RlbXAiID0gInRvbWF0byIsICAKIyAgICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKIyAgICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAojICAgICAidmFyX3RlbXAiID0gImRhcmtnb2xkZW5yb2QxIgojICAgKSkgKyAKIyAgIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAojICAgZ2d0aXRsZSgiRWlnaHQgV2Vla3MiKSArIAojICAgbGFicyh5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwKIyAgICAgICAgeCA9ICIiKSArIAojICAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsgCiMgICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAojICAgICAgICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSAyNzAsIGhqdXN0ID0gMCwgdmp1c3QgPSAwLjUpKQojIApnZ2FycmFuZ2UoZGFpbHlfcGxvdCwgd2Vla19wbG90LCBmb3VyX3dlZWtfcGxvdCwgCiAgICAgICAgICBjb21tb24ubGVnZW5kID0gVCwgbnJvdyA9IDEsIGxhYmVscyA9ICJBVVRPIiwgbGVnZW5kID0gImJvdHRvbSIpCmBgYAoKYGBge3IgbWlzYy1jb3JyLXBlcmlvZHMsIGluY2x1ZGUgPSBGfQojVGhlIGRpZmZlcmVudCB0aW1lIHBlcmlvZHMgZXhhbWluZWQgYnkgdGhpcyBjbGltYXRlIGRhdGEgaGlnaGxpZ2h0cyB0aGF0IHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBtaW5pbXVtIGFuZCBtYXhpbXVtIHRlbXBlcmF0dXJlcyBjaGFuZ2VzIGJhc2VkIG9uIHRoZSB3aW5kb3cgZXhhbWluZWQuIEZvciBleGFtcGxlLCBtaW5pbXVtIGFuZCBtYXhpbXVtIHRlbXBlcmF0dXJlcyBleHBlcmllbmNlZCBvdmVyIHdlZWtseSBpbnRlcnZhbHMgYXJlIGNsb3NlbHkgbGlua2VkLCB3aGVyZWFzIHRoZXJlIGlzIGEgZGlzdGluY3Qgc2Vhc29uYWwgY3ljbGUgaW4gdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG1pbmltdW0gYW5kIG1heGltdW0gdGVtcGVyYXR1cmVzIGV4cGVyaWVuY2VkIG92ZXIgcGVyaW9kcyBvZiBmb3VyIHdlZWtzLiAKCm9uZV93ZWVrX2RveV9kYXRhID0gd2Vla190ZW1wcyAlPiUgCiAgbXV0YXRlKGRveSA9IHlkYXkoZGF0ZSkpCgpvbmVfd2Vla190ZW1wX2NpcmNsZSA9IGdncGxvdChvbmVfd2Vla19kb3lfZGF0YSwgYWVzKHggPSBzZXZlbl9kYXlfbWVhbl9tYXgsIHkgPSBzZXZlbl9kYXlfbWVhbl9taW4sIGNvbG91ciA9IGRveSkpICsgCiAgZ2VvbV9wb2ludCgpICsgCiAgc2NhbGVfY29sb3VyX2dyYWRpZW50MigKICAgIGhpZ2ggPSAiZG9kZ2VyYmx1ZTQiLAogICAgbWlkID0gImNvcmFsMiIsCiAgICBsb3cgPSAiZG9kZ2VyYmx1ZTQiLAogICAgbWlkcG9pbnQgPSAxODIuNSkgKyAKICBsYWJzKHggPSAiTWF4LiBUZW1wLiAowrBDKSIsCiAgICAgICB5ID0gIk1pbi4gVGVtcC4gKMKwQykiKSArIAogIGxhYnMoeCA9ICJNYXguIFRlbXAuICjCsEMpIiwKICAgICAgIHkgPSAiTWluLiBUZW1wLiAowrBDKSIpICsgCiAgZ2d0aXRsZSgiT25lIFdlZWsiKSArIAogIHRoZW1lX21hdHQoKQoKZm91cl93ZWVrX2RveV9kYXRhID0gZm91cl93ZWVrX3RlbXBzICU+JSAKICBtdXRhdGUoZG95ID0geWRheShkYXRlKSkKCmZvdXJfd2Vla190ZW1wX2NpcmNsZSA9IGdncGxvdChmb3VyX3dlZWtfZG95X2RhdGEsIGFlcyh4ID0gYHR3ZW50eS1laWdodF9kYXlfbWF4YCwgeSA9IGB0d2VudHktZWlnaHRfZGF5X21pbmAsIGNvbG91ciA9IGRveSkpICsgCiAgZ2VvbV9wb2ludCgpICsgCiAgc2NhbGVfY29sb3VyX2dyYWRpZW50MigKICAgIGhpZ2ggPSAiZG9kZ2VyYmx1ZTQiLAogICAgbWlkID0gImNvcmFsMiIsCiAgICBsb3cgPSAiZG9kZ2VyYmx1ZTQiLAogICAgbWlkcG9pbnQgPSAxODIuNSkgKyAKICBsYWJzKHggPSAiTWF4LiBUZW1wLiAowrBDKSIsCiAgICAgICB5ID0gIk1pbi4gVGVtcC4gKMKwQykiKSArIAogIGdndGl0bGUoIkZvdXIgV2VlayIpICsgCiAgdGhlbWVfbWF0dCgpCgpnZ2FycmFuZ2Uob25lX3dlZWtfdGVtcF9jaXJjbGUsIGZvdXJfd2Vla190ZW1wX2NpcmNsZSwKICAgICAgICAgIGNvbW1vbi5sZWdlbmQgPSBULCBsYWJlbHMgPSAiQVVUTyIsIGxlZ2VuZCA9ICJib3R0b20iKQpgYGAKCk9yZ2FuaXNtcyBhcmUgdW5saWtlbHkgdG8gYWNjbGltYXRlIGluc3RhbnRhbmVvdXNseSB0byBjaGFuZ2VzIGluIHRlbXBlcmF0dXJlLiBUbyBleHBsb3JlIHRoZSBwb3RlbnRpYWwgdGVtcG9yYWwgd2luZG93IHRoZXNlIGNvcGVwb2RzIGFyZSByZXNwb25kaW5nIHRvLCB3ZSBleGFtaW5lZCB0aGUgY29ycmVsYXRpb24gYmV0d2VlbiB0aGVybWFsIGxpbWl0cyBhbmQgc3VtbWFyaWVzIG9mIHRoZSB0aGVybWFsIGVudmlyb25tZW50IG92ZXIgZGlmZmVyZW50IHBlcmlvZHMgb2YgdGltZS4gRm9yIGVhY2ggc3BlY2llcyAoaW5jbHVzaXZlIG9mIGFsbCBzZXhlcyBhbmQgc3RhZ2VzKSwgd2UgZXhhbWluZWQgdGhlIGNvcnJlbGF0aW9uIGJldHdlZW4gQ1RtYXggYW5kIG9uZSBvZiBuaW5lIHJlcHJlc2VudGF0aW9ucyBvZiB0aGUgdGhlcm1hbCBlbnZpcm9ubWVudCBjYWxjdWxhdGVkIGZvciBwZXJpb2RzIG9mIHRpbWUgZnJvbSAxIHRvIDYwIGRheXMgYmVmb3JlIGNvbGxlY3Rpb24uIFRoZXNlIHBhcmFtZXRlcnMgaW5jbHVkZSB0aGUgb3ZlcmFsbCBtYXhpbXVtLCBtaW5pbXVtLCBtZWRpYW4sIGFuZCBtZWFuIHRlbXBlcmF0dXJlIGZvciB0aGUgcGVyaW9kIG9mIHRpbWUsIHRoZSB0ZW1wZXJhdHVyZSByYW5nZSBhbmQgdmFyaWFuY2UgZHVyaW5nIHRoaXMgdGltZSwgYW5kIHRoZSBtZWFuIGRhaWx5IHRlbXBlcmF0dXJlIG1heGltdW0sIG1pbmltdW0sIGFuZCByYW5nZS4gV2UgYWxzbyBleGFtaW5lZCB0aGUgY29ycmVsYXRpb24gYmV0d2VlbiBDVG1heCBhbmQgdGhlIHRlbXBlcmF0dXJlIHJlY29yZGVkIGF0IHRoZSB0aW1lIG9mIGNvbGxlY3Rpb24uIAoKYGBge3IgZXN0aW1hdGluZy1wcmVkaWN0b3ItY29ycmVsYXRpb25zLCBpbmNsdWRlID0gRn0KI1dlIGNhbiBzZWUgdGhhdCwgaW4gZ2VuZXJhbCwgY29wZXBvZHMgYXJlIHJlc3BvbmRpbmcgdG8gcHJveGltYXRlIGN1ZXMgZnJvbSB0aGUgdGhlcm1hbCBlbnZpcm9ubWVudCwgd2l0aCBjb3JyZWxhdGlvbnMgZ2VuZXJhbGx5IGRyb3BwaW5nIG9mZiBzdWJzdGFudGlhbGx5IGFzIGFjY2xpbWF0aW9uIHdpbmRvdyBkdXJhdGlvbiBpbmNyZWFzZXMuIEFuIGV4Y2VwdGlvbiBpcyAqRXBpc2NodXJhIGxhY3VzdHJpcyosIHdoaWNoIGFwcGVhcnMgdG8gYmUgcmVzcG9uZGluZyB0byBtYXhpbXVtIHRlbXBlcmF0dXJlcyBleHBlcmllbmNlZCBvdmVyIGEgMjAgZGF5IHRpbWUgcGVyaW9kLiAKCiMjIyBQdWxsaW5nIHByZWRpY3RvcnMgYW5kIG1lYXN1cmluZyBjb3JyZWxhdGlvbnMgZm9yIG11Y2ggZmluZXIgdGltZXNjYWxlczsgMS01NiBkYXlzCgpudW1fY29sbHMgPSBmdWxsX2RhdGEgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBzZWxlY3QoY29sbGVjdGlvbl9kYXRlLCBzcF9uYW1lKSAlPiUgIAogIGRpc3RpbmN0KCkgJT4lICAKICBjb3VudChzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKG4gPj0gNSkKCmNvcnJfdmFscyA9IGRhdGEuZnJhbWUoKQoKZHVyX3ZhbHMgPSBjKDE6NTApCmZvcihpIGluIGR1cl92YWxzKXsKICAKICBkdXJhdGlvbl90ZW1wcyA9IGdldF9wcmVkaWN0b3JzKGRhaWx5X3ZhbHVlcyA9IGRhaWx5X3RlbXBfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHNlbGVjdCh0ZW1wX2RhdGEsIC1ob3VyKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSBpKSAlPiUgCiAgICBmaWx0ZXIoZGF0ZSAlaW4lIGFzX2RhdGUodW5pcXVlKGZ1bGxfZGF0YSRjb2xsZWN0aW9uX2RhdGUpKSkKICAKICBjb3JyX2RhdGEgPSBmdWxsX2RhdGEgJT4lCiAgICBzZWxlY3Qoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXAsIHNleCwgY3RtYXgpICU+JSAKICAgIGZpbHRlcihzcF9uYW1lICVpbiUgbnVtX2NvbGxzJHNwX25hbWUpICU+JSAKICAgICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgICBtdXRhdGUoY29sbGVjdGlvbl9kYXRlID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpKSAlPiUgCiAgICBpbm5lcl9qb2luKGR1cmF0aW9uX3RlbXBzLCBqb2luX2J5KGNvbGxlY3Rpb25fZGF0ZSA9PSBkYXRlKSkgJT4lIAogICAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKGNvbGxlY3Rpb25fdGVtcCwgY29udGFpbnMoImRheV8iKSksCiAgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInZhbHVlIiwgCiAgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicHJlZGljdG9yIikgJT4lICAKICAgIGdyb3VwX2J5KHNwX25hbWUsIHByZWRpY3RvcikgJT4lIAogICAgc3VtbWFyaXNlKGNvcnJlbGF0aW9uID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRlc3RpbWF0ZSwKICAgICAgICAgICAgICBwLnZhbHVlID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRwLnZhbHVlLAogICAgICAgICAgICAgIGNpX2xvdyA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkY29uZi5pbnRbMV0sCiAgICAgICAgICAgICAgY2lfaGlnaCA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkY29uZi5pbnRbMl0sCiAgICAgICAgICAgICAgLmdyb3VwcyA9ICJrZWVwIikgJT4lIAogICAgZmlsdGVyKHByZWRpY3RvciAhPSAiY29sbGVjdGlvbl90ZW1wIikgJT4lIAogICAgbXV0YXRlKHNpZyA9IGlmZWxzZShwLnZhbHVlIDwwLjA1LCAiU2lnLiIsICJOb24gU2lnLiIpKSAlPiUgCiAgICBzZXBhcmF0ZShwcmVkaWN0b3IsICJfZGF5XyIsIGludG8gPSBjKE5BLCAicGFyYW1ldGVyIikpICU+JSAKICAgIG11dGF0ZShkdXJhdGlvbiA9IGkpCiAgCiAgY29ycl92YWxzID0gYmluZF9yb3dzKGNvcnJfdmFscywgY29ycl9kYXRhKQp9Cgpjb2xsX2NvcnIgPSBmdWxsX2RhdGEgJT4lCiAgZmlsdGVyKHNwX25hbWUgJWluJSBudW1fY29sbHMkc3BfbmFtZSkgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgc3VtbWFyaXNlKGNvcnJlbGF0aW9uID0gY29yLnRlc3QoY3RtYXgsIGNvbGxlY3Rpb25fdGVtcCkkZXN0aW1hdGUsCiAgICAgICAgICAgIHAudmFsdWUgPSBjb3IudGVzdChjdG1heCwgY29sbGVjdGlvbl90ZW1wKSRwLnZhbHVlLAogICAgICAgICAgICBjaV9sb3cgPSBjb3IudGVzdChjdG1heCwgY29sbGVjdGlvbl90ZW1wKSRjb25mLmludFsxXSwKICAgICAgICAgICAgY2lfaGlnaCA9IGNvci50ZXN0KGN0bWF4LCBjb2xsZWN0aW9uX3RlbXApJGNvbmYuaW50WzJdKSAlPiUgCiAgbXV0YXRlKHNpZyA9IGlmZWxzZShwLnZhbHVlIDwwLjA1LCAiU2lnLiIsICJOb24gU2lnLiIpKSAlPiUgCiAgbXV0YXRlKGR1cmF0aW9uID0gMCwKICAgICAgICAgcGFyYW1ldGVyID0gImNvbGxfdGVtcCIpCgpjb3JyX3ZhbHMgPSBjb3JyX3ZhbHMgJT4lICAKICBtdXRhdGUoZHVyYXRpb24gPSBhcy5udW1lcmljKGR1cmF0aW9uKSkgJT4lIAogIGJpbmRfcm93cyhjb2xsX2NvcnIpCgpgYGAKClNob3duIGJlbG93IGFyZSB0aGUgY29ycmVsYXRpb24gY29lZmZpY2llbnRzIGZvciB0aGVzZSByZWxhdGlvbnNoaXBzLiBFYWNoIGZhY2V0IHNob3dzIHRoZSByZWxhdGlvbnNoaXAgZm9yIGEgZGlmZmVyZW50IHBhcmFtZXRlciwgcGxvdHRlZCBhZ2FpbnN0IHRoZSBkdXJhdGlvbiBvZiB0aGUgdGltZSBwZXJpb2QgYmVmb3JlIGNvbGxlY3Rpb24uIAoKYGBge3Igc3VwcC1maWctY29ycmVsYXRpb24tZHVyYXRpb25zLCBmaWcud2lkdGg9MTQsIGZpZy5oZWlnaHQ9OCwgaW5jbHVkZSA9IFR9CmNvcnJfdmFscyAlPiUgCiAgbXV0YXRlKHBhcmFtZXRlciA9IGZjdF9yZWxldmVsKHBhcmFtZXRlciwgYygibWluIiwgIm1heCIsICJyYW5nZSIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAibWVhbiIsICJtZWQiLCAidmFyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJtZWFuX21pbiIsICJtZWFuX21heCIsICJtZWFuX3JhbmdlIikpKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gZHVyYXRpb24sIHkgPSBjb3JyZWxhdGlvbiwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcCgufnBhcmFtZXRlcikgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAwKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDAuOSkgKyAKICBnZW9tX2xpbmUobGluZXdpZHRoID0gMS41KSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGxhYnMoeCA9ICJEdXJhdGlvbiAoZGF5cykiLAogICAgICAgeSA9ICJDb3JyZWxhdGlvbiIsIAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHRoZW1lX21hdHRfZmFjZXRzKCkKYGBgCgpUaGlzIHRhYmxlIGNvbnRhaW5zIHRoZSB0b3AgdGhyZWUgZmFjdG9ycyBmb3IgZWFjaCBzcGVjaWVzIChiYXNlZCBvbiBjb3JyZWxhdGlvbiBjb2VmZmljaWVudCkuCgpgYGB7ciBwcmVkaWN0b3ItY29ycmVsYXRpb25zLCBpbmNsdWRlID0gRn0KY29ycl92YWxzICU+JSAgCiAgZmlsdGVyKHNpZyA9PSAiU2lnLiIpICU+JSAKICBkcm9wX25hKGNvcnJlbGF0aW9uKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lCiAgYXJyYW5nZShkZXNjKGNvcnJlbGF0aW9uKSkgJT4lIAogIHNsaWNlX2hlYWQobiA9IDMpICU+JSAKICBzZWxlY3QoIlNwZWNpZXMiID0gc3BfbmFtZSwgIlByZWRpY3RvciIgPSBwYXJhbWV0ZXIsICJEdXJhdGlvbiIgPSBkdXJhdGlvbiwgIkNvcnJlbGF0aW9uIiA9IGNvcnJlbGF0aW9uLCAiUC1WYWx1ZSIgPSBwLnZhbHVlKSAlPiUgCiAga25pdHI6OmthYmxlKGFsaWduID0gImMiKQpgYGAKClNob3duIGhlcmUgaXMgYSBncmFwaGljYWwgc3VtbWFyeSBvZiB0aGUgZHVyYXRpb24gb2YgdGhlIGJlc3QgcHJlZGljdG9ycyBmb3IgZWFjaCBzcGVjaWVzLiBOb3RlIHRoYXQgZm9yIHRoZSB0d28gTGVwdG9kaWFwdG9taWRzLCBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlIGhhZCB0aGUgbGFyZ2VzdCBjb3JyZWxhdGlvbiBjb2VmZmljaWVudCBzbyBkdXJhdGlvbiBpcyB6ZXJvLiBUaGlzIHJlcHJlc2VudGF0aW9uIGhpZ2hsaWdodHMgdGhhdCB0aGVyZSBpcyB2YXJpYXRpb24gYWNyb3NzIHRoZSBjb21tdW5pdHkgbm90IG9ubHkgaW4gdGhlIHBvdGVudGlhbCBkcml2ZXIgKGUuZy4gbWluaW11bSB2cy4gbWF4aW11bSB0ZW1wZXJhdHVyZXMpIGJ1dCBhbHNvIGluIHRoZSBkdXJhdGlvbiBvZiB0aW1lLiBUaGlzIHZhcmlhdGlvbiBpcyBub3QgZ3JvdXBlZCBieSBzZWFzb24gKHRoZSB3aW50ZXIgYW5kIHN1bW1lciBjb21tdW5pdGllcyBib3RoIGhhdmUgcmVwcmVzZW50YXRpdmUgc3BlY2llcyBhcHBhcmVudGx5IHJlc3BvbmRpbmcgdG8gc2hvcnQgYW5kIGxvbmcgZHVyYXRpb25zKS4gCgpgYGB7ciBtYWluLWZpZy1hY2MtZHVyYXRpb25zfQpkdXJhdGlvbl9wbG90ID0gY29ycl92YWxzICU+JSAgCiAgZmlsdGVyKHNpZyA9PSAiU2lnLiIpICU+JSAKICBkcm9wX25hKGNvcnJlbGF0aW9uKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lCiAgYXJyYW5nZShkZXNjKGNvcnJlbGF0aW9uKSkgJT4lIAogIHNsaWNlX2hlYWQobiA9IDEpICU+JSAKICB1bmdyb3VwKCkgJT4lIAogIG11dGF0ZSgibnVtIiA9IHJvd19udW1iZXIoKSwgCiAgICAgICAgIHNwX25hbWUgPSBmY3RfcmVvcmRlcihzcF9uYW1lLCBkdXJhdGlvbiwgLmZ1biA9IG1lYW4sIC5kZXNjID0gVCkpICU+JSAKICBhcnJhbmdlKHNwX25hbWUpICU+JSAKICBzZWxlY3QoIlNwZWNpZXMiID0gc3BfbmFtZSwgIlByZWRpY3RvciIgPSBwYXJhbWV0ZXIsICJEdXJhdGlvbiIgPSBkdXJhdGlvbiwgIkNvcnJlbGF0aW9uIiA9IGNvcnJlbGF0aW9uLCBudW0pICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBTcGVjaWVzLCB5ID0gRHVyYXRpb24sIGZpbGwgPSBQcmVkaWN0b3IsIGdyb3VwID0gbnVtKSkgKyAKICBnZW9tX2JhcihzdGF0ID0gImlkZW50aXR5Iiwgd2lkdGggPSAwLjUsIHBvc2l0aW9uID0gcG9zaXRpb25fZG9kZ2Uod2lkdGggPSAwLjYpLAogICAgICAgICAgIGNvbG91ciA9ICJibGFjayIpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gYygiY29sbF90ZW1wIiA9ICJibGFjayIsICJtYXgiID0gIndoaXRlIiwgIm1pbiIgPSAiZ3JleSIpKSArIAogIGxhYnMoeCA9ICIiLCAKICAgICAgIHkgPSAiRHVyYXRpb24gXG4oZGF5cykiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF9ibGFuaygpKQpgYGAKCmBgYHtyIG1haW4tZmlnLWFjYy1jb3JyZWxhdGlvbnMsIGZpZy53aWR0aD02LCBmaWcuaGVpZ2h0PTh9CmNvcnJlbGF0aW9uX2NvZWZfcGxvdCA9IGNvcnJfdmFscyAlPiUgIAogIGZpbHRlcihzaWcgPT0gIlNpZy4iIHwgcGFyYW1ldGVyID09ICJjb2xsX3RlbXAiKSAlPiUgCiAgZHJvcF9uYShjb3JyZWxhdGlvbikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JQogIGZpbHRlcihwYXJhbWV0ZXIgPT0gImNvbGxfdGVtcCIgfCBjb3JyZWxhdGlvbiA9PSBtYXgoY29ycmVsYXRpb24pKSAlPiUgCiAgYXJyYW5nZShzcF9uYW1lLCBwYXJhbWV0ZXIpICU+JSAKICBtdXRhdGUoIm51bSIgPSByb3dfbnVtYmVyKCkpICU+JSAKICBzZWxlY3QoIlNwZWNpZXMiID0gc3BfbmFtZSwgIlByZWRpY3RvciIgPSBwYXJhbWV0ZXIsICJEdXJhdGlvbiIgPSBkdXJhdGlvbiwgIkNvcnJlbGF0aW9uIiA9IGNvcnJlbGF0aW9uLCBudW0pICU+JSAKICBtdXRhdGUoUHJlZGljdG9yID0gaWZfZWxzZShQcmVkaWN0b3IgPT0gImNvbGxfdGVtcCIsIFByZWRpY3RvciwgImJlc3QiKSkgJT4lIAogIHVuZ3JvdXAoKSAlPiUgCiAgbXV0YXRlKFNwZWNpZXMgPSBmY3RfcmVvcmRlcihTcGVjaWVzLCBEdXJhdGlvbiwgLmZ1biA9IG1heCwgLmRlc2MgPSBUKSkgJT4lIAogIGdncGxvdChhZXMoeCA9IFNwZWNpZXMsIHkgPSBDb3JyZWxhdGlvbiwgZmlsbCA9IFByZWRpY3RvciwgZ3JvdXAgPSBudW0pKSArIAogIGdlb21fYmFyKHN0YXQgPSAiaWRlbnRpdHkiLCB3aWR0aCA9IDAuNSwgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDAuNiksCiAgICAgICAgICAgY29sb3VyID0gImJsYWNrIikgKyAKICBsYWJzKHkgPSAiQ29ycmVsYXRpb24gXG5Db2VmZmljaWVudCIsCiAgICAgICBmaWxsID0gIkNvcnJlbGF0ZSIpICsKICBzY2FsZV9maWxsX21hbnVhbCh2YWx1ZXMgPSBjKCJjb2xsX3RlbXAiID0gImJsYWNrIiwgImJlc3QiID0gIndoaXRlIikpICsgCiAgc2NhbGVfeV9jb250aW51b3VzKGJyZWFrcyA9IGMoMCwgMSksIGxpbWl0cyA9IGMoMCwxKSkgKwogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMzAwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKCmdnYXJyYW5nZShkdXJhdGlvbl9wbG90LCBjb3JyZWxhdGlvbl9jb2VmX3Bsb3QsIG5yb3cgPSAyLCBsZWdlbmQgPSAicmlnaHQiLCBsYWJlbHMgPSAiQVVUTyIsCiAgICAgICAgICBoZWlnaHRzID0gYygwLjQsIDAuNikpCmBgYAoKYGBge3IgbWlzYy1hY2MtZHVyYXRpb24tcGxvdCwgZmlnLndpZHRoPTcsIGZpZy5oZWlnaHQ9NCwgaW5jbHVkZSA9IEZ9CiMgUGhlbm90eXBpYyB2YXJpYXRpb24gKGxpa2UgYWNjbGltYXRpb24gb2YgdGhlcm1hbCBsaW1pdHMpIGlzIGEgcGh5c2lvbG9naWNhbCBwcm9jZXNzLiBkZXBlbmRpbmcgb24gdGhlIG1lY2hhbmlzdGljIHVuZGVycGlubmluZ3MgKGNoYW5nZXMgaW4gSFNQIGV4cHJlc3Npb24sIGV0Yy4pLCB0aGUgYW1vdW50IG9mIHRpbWUgaXQgdGFrZXMgZm9yIGFuIGluZGl2aWR1YWwgdG8gYWNjbGltYXRlIG1heSB2YXJ5IGJhc2VkIG9uIGJvZHkgc2l6ZSAobGFyZ2VyIHNwZWNpZXMsIG1vcmUgY2VsbHMsIG1vcmUgdGltZSByZXF1aXJlZCB0byBhY2NsaW1hdGUpLiBTaG93biBoZXJlIGlzIHRoZSBkdXJhdGlvbiBvZiB0aGUgZW52aXJvbm1lbnRhbCBhY2NsaW1hdGlvbiB3aW5kb3cgdGhlIGNvcGVwb2RzIGFwcGVhciB0byBiZSByZXNwb25kaW5nIHRvLiAgCgptZWFuX3NpemVzID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lICAKICBzdW1tYXJpc2UobWVhbl9zaXplID0gbWVhbihzaXplLCBuYS5ybSA9IFQpKQoKY29ycl92YWxzICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKGNvcnJlbGF0aW9uID09IG1heChjb3JyZWxhdGlvbikpICU+JSAgCiAgaW5uZXJfam9pbihtZWFuX3NpemVzLCBieSA9ICJzcF9uYW1lIikgJT4lIAogIHNlbGVjdChzcF9uYW1lLCBkdXJhdGlvbiwgbWVhbl9zaXplKSAlPiUgIAogIGdncGxvdChhZXMoeCA9IG1lYW5fc2l6ZSwgeSA9IGR1cmF0aW9uKSkgKyAKICBnZW9tX3BvaW50KGFlcyhjb2xvdXIgPSBzcF9uYW1lKSwgCiAgICAgICAgICAgICBzaXplID0gNCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiTWVhbiBGZW1hbGUgU2l6ZSAobW0pIiwKICAgICAgIHkgPSAiQWNjbGltYXRpb24gRHVyYXRpb24iLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKIyMgVHJhaXQgVmFyaWF0aW9uIAoKU2hvd24gYmVsb3cgYXJlIHRoZSBjbHV0Y2ggc2l6ZSBkaXN0cmlidXRpb25zIGZvciB0aGUgdGhyZWUgZGlhcHRvbWlpZCBzcGVjaWVzLCB3aGljaCBwcm9kdWNlIGVnZyBzYWNzIHRoYXQgYWxsb3cgZm9yIGVhc3kgcXVhbnRpZmljYXRpb24gb2YgZmVjdW5kaXR5LiAKCmBgYHtyIHN1cHAtZmlnLWZlY3VuZGl0eS1oaXN0b2dyYW0sIGZpZy53aWR0aD03LCBmaWcuaGVpZ2h0PTEwfQpmdWxsX2RhdGEgJT4lICAKICBkcm9wX25hKGZlY3VuZGl0eSkgJT4lICAKICBnZ3Bsb3QoYWVzKHggPSBmZWN1bmRpdHksIGZpbGwgPSBzcF9uYW1lX3N1YikpICsgCiAgZmFjZXRfd3JhcCgufnNwX25hbWVfc3ViLCBuY29sID0gMSkgKyAKICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDIpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGxhYnMoeCA9ICJGZWN1bmRpdHkgKCMgRWdncykiKSArCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpPbmUgb2YgdGhlIG1haW4gYWltcyBvZiB0aGlzIHByb2plY3QgaXMgdG8gZXhhbWluZSB0aGUgcGF0dGVybnMgYW5kIHByb2Nlc3NlcyBkcml2aW5nIHZhcmlhdGlvbiBpbiB1cHBlciB0aGVybWFsIGxpbWl0cyBhY3Jvc3MgdGhlc2Ugc3BlY2llcyBvZiBjb3BlcG9kcy4gCgojIyMgVmFyaWF0aW9uIHdpdGggdGVtcGVyYXR1cmUgCgpXZSBleHBlY3Qgb25lIG9mIHRoZSBwcmltYXJ5IGRyaXZlcnMgb2YgY29wZXBvZCB0aGVybWFsIGxpbWl0cyB0byBiZSB0ZW1wZXJhdHVyZSwgYXMgaW5kaXZpZHVhbHMgYWNjbGltYXRlIHRvIHNlYXNvbmFsIGNoYW5nZXMuIFNob3duIGJlbG93IGFyZSB0aGUgc2Vhc29uYWwgcGF0dGVybnMgb2Ygd2hlbiBjb3BlcG9kcyB3ZXJlIGluY2x1ZGVkIGluIENUbWF4IG1lYXN1cmVtZW50cyAoYSBwcm94eSBmb3IgdGhlIHNlYXNvbiBvZiBvY2N1cnJlbmNlKSwgYW5kIHRoZXJtYWwgbGltaXRzIGZvciBlYWNoIHNwZWNpZXMgcGxvdHRlZCBhZ2FpbnN0IHRoZSB0ZW1wZXJhdHVyZSBhdCB0aGUgdGltZSBvZiBjb2xsZWN0aW9uLiBXZSBnZW5lcmFsbHkgc2VlIGFuIGluY3JlYXNlIGluIHRoZXJtYWwgbGltaXRzIHdpdGggaW5jcmVhc2luZyBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlLgoKYGBge3IgZmlnLndpZHRoPTcsIGZpZy5oZWlnaHQ9NX0Kc3BfY3RtYXhfdGVtcCA9IGZ1bGxfZGF0YSAlPiUgCiAgZmlsdGVyKHNwX25hbWUgIT0gIk9zcGhyYW50aWN1bSBsYWJyb25lY3R1bSIpICU+JSAKICBtdXRhdGUoc3BfbmFtZSA9IGFzLmZhY3RvcihzcF9uYW1lKSwKICAgICAgICAgc3BfbmFtZSA9IGZjdF9yZW9yZGVyKHNwX25hbWUsIGN0bWF4LCAuZGVzYyA9IFQpKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gY29sbGVjdGlvbl90ZW1wLCB5ID0gY3RtYXgsIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMS41LCBjb2xvdXIgPSAiZ3JleTMwIikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuNCkgKyAKICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wLiAowrBDKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQpgYGAKClRoZSBpbnRlcmFjdGlvbiBiZXR3ZWVuIHNlYXNvbmFsIGNoYW5nZXMgaW4gdGVtcGVyYXR1cmUgYW5kIHRoZSBhY2NsaW1hdGlvbiBvZiB0aGVybWFsIGxpbWl0cyBsaWtlbHkgYWZmZWN0cyB2dWxuZXJhYmlsaXR5IG9mIGVhY2ggc3BlY2llcyB0byB3YXJtaW5nLiBTaG93biBiZWxvdyBhcmUgd2FybWluZyB0b2xlcmFuY2UgdmFsdWVzIGZvciBlYWNoIHNwZWNpZXMsIGNhbGN1bGF0ZWQgYXMgdGhlIGRpZmZlcmVuY2UgYmV0d2VlbiBpbmRpdmlkdWFsIENUbWF4IGFuZCB0aGUgdGVtcGVyYXR1cmUgYXQgdGhlIHRpbWUgb2YgY29sbGVjdGlvbi4gQWxsIHNwZWNpZXMgbWFpbnRhaW5lZCBzb21lIGRlZ3JlZSBvZiBidWZmZXIgYmV0d2VlbiBlbnZpcm9ubWVudGFsIHRlbXBlcmF0dXJlcyBhbmQgdXBwZXIgdGhlcm1hbCBsaW1pdHMsIGJ1dCAqTC4gbWludXR1cyogYXBwZWFycyB0byBhcHByb2FjaCBpdHMgdXBwZXIgdGhlcm1hbCBsaW1pdCBkdXJpbmcgdGhlIHdhcm1lc3QgY29sbGVjdGlvbnMgZHVyaW5nIHRoZSBzdW1tZXIuIAoKQWxzbyBzaG93biBiZWxvdyBpcyB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gZmVjdW5kaXR5ICh0aGUgbnVtYmVyIG9mIGVnZ3MgY29udGFpbmVkIGluIGEgY2x1dGNoKSBmb3IgdGhlIHRocmVlIGRpYXB0b21pZCBzcGVjaWVzLiBGb3IgdGhlIHR3byBMZXB0b2RpYXB0b211cyBzcGVjaWVzLCB0aGVyZSBpcyBubyByZWxhdGlvbnNoaXAgYmV0d2VlbiBjbHV0Y2ggc2l6ZSBhbmQgdGVtcGVyYXR1cmUsIHdoaWxlIHRoZXJlIGFwcGVhcnMgdG8gYmUgYSBnZW5lcmFsIGluY3JlYXNlIGluIGNsdXRjaCBzaXplIHdpdGggdGVtcGVyYXR1cmUgaW4gdGhlIFNraXN0b2RpYXB0b211cyBzcGVjaWVzLiAKCmBgYHtyIG1haW4tZmlnLXRyYWl0LWNvbGwtdGVtcC1wbG90cywgZmlnLndpZHRoPTE1LCBmaWcuaGVpZ2h0PTd9Cgp3dF90ZW1wID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc3BfbmFtZSAhPSAiT3NwaHJhbnRpY3VtIGxhYnJvbmVjdHVtIikgJT4lIAogIGdncGxvdChhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IHdhcm1pbmdfdG9sLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3BvaW50KHNpemUgPSAzLAogICAgICAgICAgICAgYWxwaGEgPSAwLjMpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgbGluZXdpZHRoID0gMykgKwogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICB5ID0gIldhcm1pbmcgVG9sZXJhbmNlICjCsEMpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgICsgCiAgeWxpbSgwLDMwKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKCmVnZ3NfdGVtcCA9IGZ1bGxfZGF0YSAlPiUgCiAgZmlsdGVyKHNwX25hbWUgIT0gIk9zcGhyYW50aWN1bSBsYWJyb25lY3R1bSIpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBjb2xsZWN0aW9uX3RlbXAsIHkgPSBmZWN1bmRpdHksIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDMsCiAgICAgICAgICAgICBhbHBoYSA9IDAuMykgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBsaW5ld2lkdGggPSAzKSArCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcGVyYXR1cmUgKMKwQykiLCAKICAgICAgIHkgPSAiRmVjdW5kaXR5ICgjIEVnZ3MpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmdnYXJyYW5nZSh3dF90ZW1wLCBlZ2dzX3RlbXAsIAogICAgICAgICAgY29tbW9uLmxlZ2VuZCA9IFQsIGxlZ2VuZCA9ICJyaWdodCIsIGxhYmVscyA9ICJBVVRPIikKYGBgCgpgYGB7ciBzdXBwLWZpZy1sc2ljLW1vcnBocywgaW5jbHVkZSA9IEZ9Cm1vcnBoX2RhdGEgPSBmdWxsX2RhdGEgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIgJiBzcGVjaWVzID09ICJsZXB0b2RpYXB0b211c19zaWNpbGlzIikgJT4lICBtdXRhdGUoc3BfbmFtZSA9IGNhc2Vfd2hlbigKICAgIHNwX25hbWUgPT0gIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMiICYgc2l6ZSA+PSAwLjg5IH4gIkxhcmdlIiwKICAgIHNwX25hbWUgPT0gIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMiICYgc2l6ZSA8IDAuODkgfiAiU21hbGwiLAogICAgLmRlZmF1bHQgPSBzcF9uYW1lCiAgKSkKCmdncGxvdChtb3JwaF9kYXRhLCBhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuOCkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IFQsIGxpbmV3aWR0aCA9IDIpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcC4gKMKwQykiLCAKICAgICAgIHkgPSAiQ1RtYXggKMKwQykiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpgYGB7ciBtaXNjLWN0bWF4LXJhbmdlLXBsb3QsIGluY2x1ZGUgPSBGfQpmdWxsX2RhdGEgJT4lICAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSwgY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXApICU+JSAgCiAgc3VtbWFyaXNlKCJjdG1heF9yYW5nZSIgPSBtYXgoY3RtYXgpIC0gbWluKGN0bWF4KSwKICAgICAgICAgICAgImN0bWF4X3ZhciIgPSB2YXIoY3RtYXgpLAogICAgICAgICAgICAic2FtcGxlX3NpemUiID0gbigpKSAlPiUgCiAgdW5ncm91cCgpICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKHNwX25hbWUgIT0gIkxlcHRvZG9yYSBraW5kdGkiKSAlPiUgCiAgZmlsdGVyKHNhbXBsZV9zaXplID4gMykgJT4lIAogIGdncGxvdChhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IGN0bWF4X3ZhciwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4pICsgCiAgZ2VvbV9wb2ludCgpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgY29sb3VyID0gImJsYWNrIikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpCmBgYAoKYGBge3IgY3RtYXgtY29sbC10ZW1wLW1vZGVsLCBpbmNsdWRlID0gRn0KIyBhZHVsdF9kYXRhID0gZnVsbF9kYXRhICU+JSAKIyAgIGZpbHRlcihzZXggPT0gImZlbWFsZSIpCgojIAojIGN0bWF4X3RlbXAubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgY3RtYXggfiB0ZW1wX2NlbnQgKiBzcF9uYW1lKQojIHNpemVfdGVtcC5tb2RlbCA9IGxtKGRhdGEgPSBtb2RlbF9kYXRhLCBzaXplIH4gdGVtcF9jZW50ICogc3BfbmFtZSkKIyBmZWN1bmRfdGVtcC5tb2RlbCA9IGxtKGRhdGEgPSBkcm9wX25hKG1vZGVsX2RhdGEsIGZlY3VuZGl0eSksIGZlY3VuZGl0eSB+IHRlbXBfY2VudCAqIHNwX25hbWUpCiMgCiMgZmVjdW5kaXR5X3Jlc2lkcyA9IGNiaW5kKGRyb3BfbmEobW9kZWxfZGF0YSwgZmVjdW5kaXR5KSwgImZlY3VuZGl0eV9yZXNpZHMiID0gZmVjdW5kX3RlbXAubW9kZWwkcmVzaWR1YWxzKSAlPiUgCiMgICBzZWxlY3QoY29sbGVjdGlvbl9kYXRlLCBleHBfZGF0ZSwgcmVwbGljYXRlLCBzcGVjaWVzLCB0dWJlLCBmZWN1bmRpdHlfcmVzaWRzKQojIAojIGN0bWF4X3Jlc2lkcyA9IGNiaW5kKG1vZGVsX2RhdGEsICJyZXNpZHMiID0gY3RtYXhfdGVtcC5tb2RlbCRyZXNpZHVhbHMsICJzaXplX3Jlc2lkcyIgPSBzaXplX3RlbXAubW9kZWwkcmVzaWR1YWxzKSAlPiUgCiMgICBsZWZ0X2pvaW4oZmVjdW5kaXR5X3Jlc2lkcykKCmBgYAoKYGBge3Igc3VwcC1maWctY3RtYXgtdGltZS1pbi1sYWIsIGZpZy53aWR0aD0xNSwgZmlnLmhlaWdodD0xMH0KIyBDb3BlcG9kcyBzcGVudCBzZXZlcmFsIGRheXMgaW4gbGFiIGR1cmluZyBleHBlcmltZW50cy4gU2hvd24gYmVsb3cgYXJlIHRoZSBDVG1heCByZXNpZHVhbHMgKHRha2VuIGZyb20gYSBtb2RlbCBvZiBDVG1heCBhZ2FpbnN0IGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUpIHBsb3R0ZWQgYWdhaW5zdCB0aGUgdGltZSBzcGVudCBpbiBsYWIgYmVmb3JlIG1lYXN1cmVtZW50cyB3ZXJlIG1hZGUuIEluZGl2aWR1YWwgcmVncmVzc2lvbnMgYXJlIHNob3duIGZvciB0aGUgcmVzaWR1YWxzIGFnYWluc3QgZGF5cyBpbiBsYWIgZm9yIGVhY2ggY29sbGVjdGlvbi4gV2UgY2FuIHNlZSBjbGVhcmx5IHRoYXQgdGhlcm1hbCBsaW1pdHMgYXJlIGZhaXJseSBzdGFibGUgb3ZlciB0aW1lLiAKCgojIGdncGxvdChjdG1heF9yZXNpZHMsIGFlcyh4ID0gZGF5c19pbl9sYWIsIHkgPSByZXNpZHMsIGNvbG91ciA9IHNwX25hbWUsIGdyb3VwID0gY29sbGVjdGlvbl9kYXRlKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAojICAgZ2VvbV9wb2ludChzaXplID0gNCwgYWxwaGEgPSAwLjUpICsgCiMgICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDEpICsgCiMgICAjc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IGMoMDo1KSkgKyAKIyAgIGxhYnMoeCA9ICJEYXlzIGluIGxhYiIsIAojICAgICAgICB5ID0gIkNUbWF4IFJlc2lkdWFscyIpICsgCiMgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKIyAgIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKIyAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpgYGB7ciBzdXBwLWZpZy1tb2RlbC1wZXJmb3JtYW5jZSwgZmlnLndpZHRoPTksIGZpZy5oZWlnaHQ9MTN9Cgptb2RlbF9kYXRhID0gZnVsbF9kYXRhICU+JSAgCiAgZHJvcF9uYShzaXplLCBjdG1heCkgJT4lICAKICBmaWx0ZXIoc3BfbmFtZSAhPSAiT3NwaHJhbnRpY3VtIGxhYnJvbmVjdHVtIikgJT4lIAogIG11dGF0ZSh0ZW1wX2NlbnQgPSBzY2FsZShjb2xsZWN0aW9uX3RlbXAsIGNlbnRlciA9IFQsIHNjYWxlID0gRiksCiAgICAgICAgIHNpemVfY2VudCA9IHNjYWxlKHNpemUsIGNlbnRlciA9IFQsIHNjYWxlID0gRikpCgptaW5pbWFsLm1vZGVsID0gbG1lNDo6bG1lcihkYXRhID0gbW9kZWxfZGF0YSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiBzcF9uYW1lICsgc2V4ICsgdGVtcF9jZW50ICsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoMXxkYXlzX2luX2xhYikpCgpmdWxsLm1vZGVsID0gbG1lNDo6bG1lcihkYXRhID0gbW9kZWxfZGF0YSwKICAgICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiBzcF9uYW1lKnNleCp0ZW1wX2NlbnQgKwogICAgICAgICAgICAgICAgICAgICAgICAgICgxfGRheXNfaW5fbGFiKSkKCmRyb3AxKGZ1bGwubW9kZWwsIHRlc3QgPSAiQ2hpc3EiKQoKcGVyZm9ybWFuY2U6OnRlc3RfcGVyZm9ybWFuY2UobWluaW1hbC5tb2RlbCwgZnVsbC5tb2RlbCkKcGVyZm9ybWFuY2U6OmNoZWNrX21vZGVsKGZ1bGwubW9kZWwpCgpjYXI6OkFub3ZhKGZ1bGwubW9kZWwsIHR5cGUgPSAiSUlJIikKCnNwX2N0bWF4ID0gZW1tZWFuczo6ZW1tZWFucyhmdWxsLm1vZGVsLCBzcGVjcyA9ICJzcF9uYW1lIikgJT4lIAogIGRhdGEuZnJhbWUoKSAlPiUgCiAgc2VsZWN0KHNwX25hbWUsICJzcGVjaWVzX2N0bWF4IiA9IGVtbWVhbikKCm1vZGVsX2NvZWZzID0gZW1tZWFuczo6ZW10cmVuZHMoZnVsbC5tb2RlbCwgdmFyID0gInRlbXBfY2VudCIsIHNwZWNzID0gInNwX25hbWUiKSAlPiUgCiAgZGF0YS5mcmFtZSgpICU+JSAKICBpbm5lcl9qb2luKHNwX2N0bWF4KSAKCmN0bWF4X3Jlc2lkcyA9IG1vZGVsX2RhdGEgJT4lIAogIG11dGF0ZShyZXNpZHMgPSByZXNpZHVhbHMoZnVsbC5tb2RlbCkpCgojd3JpdGUuY3N2KG1vZGVsX2NvZWZzLCAiT3V0cHV0L0RhdGEvQVJSX2RhdGEuY3N2IikKYGBgCgpgYGB7ciBtYWluLWZpZy1BUlItc3ludGgtcGxvdCwgZmlnLndpZHRoPTksIGZpZy5oZWlnaHQ9Nn0KYXJyX2NvbWJpbmVkID0gc3ludGhfYXJyICU+JQogIGZpbHRlcihtZWFzdXJlID09ICJ1cHBlciIgJiBtZWFuX2xpbSA+IDIwKSAlPiUgCiAgc2VsZWN0KCJncm91cCIgPSBnZW51cywgYXJyLCBtZWFuX2xpbSkgJT4lIAogIG11dGF0ZSgiZGF0YXNldCIgPSAic3ludGhlc2lzIikgJT4lIAogIGJpbmRfcm93cygKICAgIHNlbGVjdChtb2RlbF9jb2VmcywgImdyb3VwIiA9IHNwX25hbWUsICdhcnInID0gdGVtcF9jZW50LnRyZW5kLCAnbWVhbl9saW0nID0gc3BlY2llc19jdG1heCkKICApICU+JSAKICBtdXRhdGUoZGF0YXNldCA9IGlmX2Vsc2UoaXMubmEoZGF0YXNldCksICJuZXcgZGF0YSIsICJzeW50aGVzaXMiKSwKICAgICAgICAgZ3JvdXAgPSBmY3RfcmVvcmRlcihncm91cCwgYXJyLCAuZGVzYyA9IFQpKQoKCmdncGxvdChhcnJfY29tYmluZWQsIGFlcyh4ID0gbWVhbl9saW0sIHkgPSBhcnIpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgCiAgICAgICAgICAgICAgbGluZXdpZHRoID0gMiwgY29sb3VyID0gImdyZXkzMCIpICsgCiAgZ2VvbV9wb2ludChkYXRhID0gZmlsdGVyKGFycl9jb21iaW5lZCwgZGF0YXNldCAhPSAibmV3IGRhdGEiKSwgCiAgICAgICAgICAgICBzaXplID0gMywgY29sb3VyID0gImJsYWNrIiwgc2hhcGUgPSAxLCBzdHJva2UgPSAxLjQpICsgCiAgZ2VvbV9wb2ludChkYXRhID0gZmlsdGVyKGFycl9jb21iaW5lZCwgZGF0YXNldCA9PSAibmV3IGRhdGEiKSwKICAgICAgICAgICAgIGFlcyhjb2xvdXIgPSBncm91cCksIAogICAgICAgICAgICAgc2l6ZSA9IDQuNSkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiVGhlcm1hbCBMaW1pdCIsIAogICAgICAgeSA9ICJBUlIiLCAKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgKwogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCgpgYGAKCiMjIyBTZXggYW5kIHN0YWdlIHZhcmlhdGlvbiBpbiB0aGVybWFsIGxpbWl0cyAKUHJldmlvdXMgc2VjdGlvbnMgaGF2ZSBnZW5lcmFsbHkgbHVtcGVkIGp1dmVuaWxlLCBmZW1hbGUsIGFuZCBtYWxlIGluZGl2aWR1YWxzIHRvZ2V0aGVyLiBUaGVyZSBtYXkgYmUgaW1wb3J0YW50IHN0YWdlLSBvciBzZXgtc3BlY2lmaWMgZGlmZmVyZW5jZXMgaW4gQ1RtYXggdGhvdWdoLiBGb3IgYWxsIHNwZWNpZXMgYnV0IE9zcGhyYW50aWN1bSwgd2UgaGF2ZSBtZWFzdXJlbWVudHMgZm9yIGluZGl2aWR1YWxzIGluIGRpZmZlcmVudCBzdGFnZXMgYW5kIG9mIGRpZmZlcmVudCBzZXhlcy4gCgpgYGB7ciBzZXgtc3RhZ2UtdGFibGV9CnNleF9zYW1wbGVfc2l6ZXMgPSBmdWxsX2RhdGEgJT4lICAKICBncm91cF9ieShzcF9uYW1lLCBzZXgpICU+JSAgCiAgc3VtbWFyaXNlKG51bSA9IG4oKSkgJT4lICAKICBwaXZvdF93aWRlcihpZF9jb2xzID0gc3BfbmFtZSwKICAgICAgICAgICAgICBuYW1lc19mcm9tID0gc2V4LCAKICAgICAgICAgICAgICB2YWx1ZXNfZnJvbSA9IG51bSwKICAgICAgICAgICAgICB2YWx1ZXNfZmlsbCA9IDApICU+JSAKICBzZWxlY3QoIlNwZWNpZXMiID0gc3BfbmFtZSwgIkp1dmVuaWxlIiA9IGp1dmVuaWxlLCAiRmVtYWxlIiA9IGZlbWFsZSwgIk1hbGUiID0gbWFsZSkKCmtuaXRyOjprYWJsZShzZXhfc2FtcGxlX3NpemVzLCBhbGlnbiA9ICJjIikKYGBgCgpBY3Jvc3MgZ3JvdXAgY29tcGFyaXNvbnMgc2hvdyB0aGF0IHRoZXJlIGFyZSBnZW5lcmFsbHkgbm8gZGlmZmVyZW5jZXMgaW4gdGhlcm1hbCBsaW1pdHMgKHJlcHJlc2VudGVkIGhlcmUgYXMgdGhlIHJlc2lkdWFscyBmcm9tIGEgQ1RtYXggfiBjb2xsZWN0aW9uX3RlbXAgeCBzcGVjaWVzIGxpbmVhciByZWdyZXNzaW9uKSwgd2l0aCB0aGUgZXhjZXB0aW9uIG9mIFNlbmVjZWxsYSBtYWxlcywgd2hpY2ggbWF5IGhhdmUgbG93ZXIgdGhlcm1hbCBsaW1pdHMgKGFsdGhvdWdoIHNhbXBsZSBzaXplcyBhcmUgdmVyeSBzbWFsbCBpbiB0aGlzIGdyb3VwKS4KCmBgYHtyIHN1cHAtZmlnLWN0bWF4LXNleCwgZmlnLndpZHRoPTE1LCBmaWcuaGVpZ2h0PTh9CiMgY3RtYXhfcmVzaWRzICU+JSAKIyAgIGZpbHRlcihzcF9uYW1lICE9ICJPc3BocmFudGljdW0gbGFicm9uZWN0dW0iKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBzZXgsIHkgPSByZXNpZHMsIGNvbG91ciA9IHNwX25hbWUpKSArIAojICAgZmFjZXRfd3JhcChzcF9uYW1lfi4pICsgCiMgICBnZW9tX2ppdHRlcih3aWR0aCA9IDAuMSwgYWxwaGEgPSAwLjUpICsgCiMgICBnZW9tX2JveHBsb3Qod2lkdGggPSAwLjQsIGZpbGwgPSBOQSwgY29sb3VyID0gImJsYWNrIiwgCiMgICAgICAgICAgICAgICAgbGluZXdpZHRoID0gMSwgb3V0bGllci5jb2xvdXIgPSBOQSkgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAojICAgdGhlbWVfbWF0dF9mYWNldHMoKQpgYGAKCmBgYHtyIHRyYWl0LXZhcmlhbmNlLWNvbGwtdGVtcCwgaW5jbHVkZSA9IEZ9CiMgCiMgR2l2ZW4gdGhlIGxvbmcgZ2VuZXJhdGlvbiB0aW1lcyBvZiB0aGVzZSBjb3BlcG9kcywgZGVjcmVhc2VzIGluIHRyYWl0IHZhcmlhbmNlIG1heSBpbmRpY2F0ZSBzZWxlY3Rpb24gb3ZlciB0aGUgc2Vhc29uYWwgY3ljbGUuIFNob3duIGJlbG93IGFyZSB0aGUgdmFyaWFuY2UgaW4gb2JzZXJ2ZWQgQ1RtYXggYW5kIHNpemUsIHBsb3R0ZWQgYWdhaW5zdCBjb2xsZWN0aW9uIGRhdGUuIFZhcmlhbmNlIGRlY3JlYXNlcyBpbiAqU2tpc3RvZGlhcHRvbXVzKiwgYnV0IHRoaXMgcGF0dGVybiBpcyBkcml2ZW4gYnkgYSBzaW5nbGUgY29sbGVjdGlvbiB3aXRoIGhpZ2ggdmFyaWFuY2UgZWFybHkgaW4gdGhlIHllYXIuIFNpemUgdmFyaWFuY2UgaW5jcmVhc2VzIHNsaWdodGx5IGluICpTa2lzdG9kaWFwdG9tdXMqLiBWYXJpYW5jZSBpbiBib3RoIENUbWF4IGFuZCBzaXplIGlzIGZhaXJseSBjb25zdGFudCBpbiAqTGVwdG9kaWFwdG9tdXMgbWludXR1cyosIHRoZSBvbmx5IG90aGVyIHNwZWNpZXMgY29sbGVjdGVkIGFjcm9zcyB0aGUgZW50aXJlIHNldCBvZiBzYW1wbGVzIHRodXMgZmFyLiAKIyAKIyBnZ3Bsb3QoZHJvcF9uYShhZHVsdF9zdW1tYXJpZXMsIGN0bWF4X3ZhciksIGFlcyh4ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gY3RtYXhfdmFyLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uLCBzY2FsZXMgPSAiZnJlZV95IikgKyAKIyAgIGdlb21fcG9pbnQoc2l6ZSA9IDIpICsgCiMgICBnZW9tX3Ntb290aChzZSA9IEYpICsgCiMgICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wLiAowrBDKSIsIAojICAgICAgICB5ID0gIkNUbWF4IFZhcmlhbmNlIikgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAojICAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAojICAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gIm5vbmUiKQojIAojIGdncGxvdChkcm9wX25hKGFkdWx0X3N1bW1hcmllcywgc2l6ZV92YXIpLCBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgeSA9IHNpemVfdmFyLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAojICAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKIyAgIGdlb21fc21vb3RoKHNlID0gRikgKyAKIyAgIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXAuICjCsEMpIiwgCiMgICAgICAgIHkgPSAiU2l6ZSBWYXJpYW5jZSIpICsgCiMgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKIyAgIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKIyAgIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpgYGB7ciBzdXBwLWZpZy1tb2RlbDItcGVyZm9ybWFuY2UsIGZpZy53aWR0aD05LCBmaWcuaGVpZ2h0PTEzfQptb2RlbDJfZGF0YSA9IG1vZGVsX2RhdGEgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIsIAogICAgICAgICBwYXRob2dlbiAhPSAidW5jZXJ0YWluIiwgCiAgICAgICAgIGRldl9lZ2dzICE9ICJ1bmNlcnRhaW4iLCAKICAgICAgICAgbGlwaWRzICE9ICJ1bmNlcnRhaW4iKSAlPiUgCiAgbXV0YXRlKHBhdGhvZ2VucyA9IGZjdF9yZWxldmVsKHBhdGhvZ2VuLCAibm8iLCAic3BvdCIsICJjbG91ZHkiLCAib3RoZXIiKSkKCm90aGVyX2ZhY3Rvcl9tb2RlbCA9IGxtZXIoZGF0YSA9IG1vZGVsMl9kYXRhLCAKICAgICAgICAgICAgICAgICAgICAgICAgICBjdG1heH5zcF9uYW1lICogY29sbGVjdGlvbl90ZW1wICsgZGV2X2VnZ3MgKyBwYXRob2dlbiArIGxpcGlkcyArICgxfGRheXNfaW5fbGFiKSkKCmRyb3AxKG90aGVyX2ZhY3Rvcl9tb2RlbCwgc2NvcGUgPSB+LiwgdGVzdCA9ICJDaGlzcSIpCgpyZWR1Y2VkX2ZhY3RvcnNfbW9kZWwgPSBsbWVyKGRhdGEgPSBtb2RlbDJfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgY3RtYXh+c3BfbmFtZSAqIGNvbGxlY3Rpb25fdGVtcCArIHBhdGhvZ2VuICsgKDF8ZGF5c19pbl9sYWIpKQoKcGVyZm9ybWFuY2U6OmNoZWNrX21vZGVsKHJlZHVjZWRfZmFjdG9yc19tb2RlbCkKCmNhcjo6QW5vdmEocmVkdWNlZF9mYWN0b3JzX21vZGVsLCB0eXBlID0gIklJSSIpCmBgYAoKYGBge3Igc3VwcC1maWctcGF0aG9nZW4tZWZmZWN0LCBmaWcud2lkdGg9OCwgZmlnLmhlaWdodD01fQplbW1lYW5zOjplbW1lYW5zKHJlZHVjZWRfZmFjdG9yc19tb2RlbCwgc3BlYyA9ICJwYXRob2dlbiIpICU+JSBlbW1lYW5zOjpjb250cmFzdChtZXRob2Q9InRydC52cy5jdHJsIixyZWY9Im5vIikgJT4lIHBsb3QoKSArIAogIGdlb21fdmxpbmUoeGludGVyY2VwdCA9IDApICsgCiAgbGFicyh4ID0gIkRpZmZlcmVuY2UgKMKwQykiLCAKICAgICAgIHkgPSAiQ29tcGFyaXNvbiIpICsgCiAgICB0aGVtZV9tYXR0KCkKYGBgCgoKCiMjIyBUcmFpdCBDb3JyZWxhdGlvbnMgYW5kIFRyYWRlLW9mZnMKCkEgcmVsYXRpb25zaGlwIGJldHdlZW4gc2l6ZSBhbmQgdXBwZXIgdGhlcm1hbCBsaW1pdHMgaGFzIGJlZW4gc3VnZ2VzdGVkIGluIGEgd2lkZSByYW5nZSBvZiBvdGhlciB0YXhhLiBTaG93biBiZWxvdyBhcmUgdGhlIG1lYXN1cmVkIHVwcGVyIHRoZXJtYWwgbGltaXRzIHBsb3R0ZWQgYWdhaW5zdCBwcm9zb21lIGxlbmd0aC4gVGhlIG92ZXJhbGwgcmVsYXRpb25zaGlwIChpbmNsdXNpdmUgb2YgYWxsIHNwZWNpZXMpIGlzIHNob3duIGFzIHRoZSBibGFjayBsaW5lIGluIHRoZSBiYWNrZ3JvdW5kLiBSZWdyZXNzaW9ucyBmb3IgZWFjaCBpbmRpdmlkdWFsIHNwZWNpZXMgYXJlIGFsc28gc2hvd24uIEFjcm9zcyB0aGUgZW50aXJlIGFzc2VtYmxhZ2UsIHRoZXJlIGlzIGEgc3Ryb25nIGRlY3JlYXNlIGluIHRoZXJtYWwgbGltaXRzIHdpdGggaW5jcmVhc2luZyBzaXplLiAgCgpgYGB7ciBtaXNjLWN0bWF4LXNpemUsIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD03fQpmdWxsX2RhdGEgJT4lIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgIAogIGdncGxvdCggYWVzKHggPSBzaXplLCB5ID0gY3RtYXgsIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDIsIGFscGhhID0gMC4zKSArIAogIGdlb21fc21vb3RoKGRhdGEgPSBmdWxsX2RhdGEsIAogICAgICAgICAgICAgIGFlcyh4ID0gc2l6ZSwgeSA9IGN0bWF4KSwKICAgICAgICAgICAgICBtZXRob2QgPSAibG0iLCAKICAgICAgICAgICAgICBjb2xvdXIgPSJibGFjayIsIAogICAgICAgICAgICAgIGxpbmV3aWR0aCA9IDIuNSkgKyAKICBsYWJzKHggPSAiTGVuZ3RoIChtbSkiLCAKICAgICAgIHkgPSAiQ1RtYXggKMKwQykiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCgpgYGAKClNob3duIGhlcmUgaXMgdGhlIHJlbGF0aW9uc2hpcCBmb3IgZWFjaCBzcGVjaWVzIGluZGl2aWR1YWxseS4gCgpgYGB7ciBtaXNjLWluZC1zcC1jdG1heC1zaXplLCBmaWcud2lkdGg9OSwgZmlnLmhlaWdodD02fQpmdWxsX2RhdGEgJT4lIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JSBmaWx0ZXIobigpID4yKSAlPiUgZmlsdGVyKCFzdHJfZGV0ZWN0KHNwX25hbWUsIHBhdHRlcm4gPSAia2luZHRpIikpICU+JSAKICBnZ3Bsb3QoIGFlcyh4ID0gc2l6ZSwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+Liwgc2NhbGVzID0gImZyZWUiLCBucm93ID0gMikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuOCkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDIpICsgCiAgbGFicyh4ID0gIkxlbmd0aCAobW0pIiwgCiAgICAgICB5ID0gIkNUbWF4ICjCsEMpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKU2hvd24gYmVsb3cgaXMgdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG1lYW4gc2l6ZSBhbmQgbWVhbiB0aGVybWFsIGxpbWl0cyBmb3IgZmVtYWxlcyBvZiBlYWNoIHNwZWNpZXMuIFdlIHNlZSB0aGF0IGxhcmdlciBzcGVjaWVzIHdpdGhpbiB0aGUgY29tbXVuaXR5IHRlbmQgdG8gaGF2ZSBhIGxvd2VyIHRoZXJtYWwgbGltaXQgdGhhbiBzbWFsbGVyIHNwZWNpZXMuIAoKYGBge3IgbWFpbi1maWctbWVhbi1jdG1heC1tZWFuLXNpemUtcGxvdCwgZmlnLndpZHRoPTksIGZpZy5oZWlnaHQ9NX0KZnVsbF9kYXRhICU+JSAKICBncm91cF9ieShzcF9uYW1lLCBzZXgpICU+JSAKICBzdW1tYXJpemUobWVhbl9jdG1heCA9IG1lYW4oY3RtYXgsIG5hLnJtID0gVCksCiAgICAgICAgICAgIG1lYW5fc2l6ZSA9IG1lYW4oc2l6ZSwgbmEucm0gPSBUKSkgJT4lIAogICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gbWVhbl9zaXplLCB5ID0gbWVhbl9jdG1heCkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyLCBjb2xvdXIgPSAiYmxhY2siKSArIAogIGdlb21fcG9pbnQoYWVzKGNvbG91ciA9IHNwX25hbWUsIHNoYXBlID0gc2V4KSwKICAgICAgICAgICAgIHNpemUgPSA1KSArIAogIGxhYnMoeCA9ICJMZW5ndGggKG1tKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpgYGB7ciBjdG1heHJlc2lkcy1zaXplLCBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9NywgaW5jbHVkZSA9IEZ9CiMgY3RtYXhfcmVzaWRzICU+JSAKIyAgICNmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBzaXplLCB5ID0gcmVzaWRzLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKIyAgIGZhY2V0X3dyYXAoc3BfbmFtZX4uKSArIAojICAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKIyAgIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMikgKyAKIyAgIGxhYnMoeCA9ICJMZW5ndGggKG1tKSIsIAojICAgICAgICB5ID0gIkNUbWF4ICjCsEMpIikgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAojICAgdGhlbWVfbWF0dCgpICsgCiMgICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKU2hvd24gaGVyZSBpcyB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gZmVjdW5kaXR5IGFuZCBzaXplLCBzaG93aW5nIHRoZSBjbGFzc2ljIHBhdHRlcm4gb2YgaW5jcmVhc2luZyBlZ2cgcHJvZHVjdGlvbiB3aXRoIGluY3JlYXNpbmcgc2l6ZS4gCgpgYGB7ciBmZWN1bmRpdHktc2l6ZX0Kc2l6ZV9mZWN1bmRfcGxvdCA9IGN0bWF4X3Jlc2lkcyAlPiUgIAogIGRyb3BfbmEoZmVjdW5kaXR5KSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gc2l6ZSwgeSA9IGZlY3VuZGl0eSwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDIsIGFscGhhID0gMC41KSArIAogIGxhYnMoeCA9ICJQcm9zb21lIGxlbmd0aCAobW0pIiwgCiAgICAgICB5ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpJbmRpdmlkdWFscyBtYXkgYWxzbyBhbGxvY2F0ZSBlbmVyZ3kgdG8gZGlmZmVyZW50IGZpdG5lc3MgcmVsYXRlZCB0cmFpdHMsIHByaW9yaXRpemluZyByZXByb2R1Y3RpdmUgb3V0cHV0IG92ZXIgZW52aXJvbm1lbnRhbCB0b2xlcmFuY2UsIGZvciBleGFtcGxlLiBTaG93biBiZWxvdyBpcyB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gQ1RtYXggcmVzaWR1YWxzIChhZ2FpbiwgY29udHJvbGxpbmcgZm9yIHRoZSBlZmZlY3RzIG9mIGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUpIGFnYWluc3QgZmVjdW5kaXR5LiBXZSBjYW4gc2VlIGNsZWFybHkgdGhhdCBpbmRpdmlkdWFscyB3aXRoIGluY3JlYXNlZCBmZWN1bmRpdHkgYXJlIG5vdCBkZWNyZWFzaW5nIHRoZXJtYWwgbGltaXRzLCBzdWdnZXN0aW5nIHRoYXQgdGhlcmUgaXMgbm8gZW5lcmdldGljIHRyYWRlLW9mZiBiZXR3ZWVuIHRoZXNlIHRyYWl0cy4gCmBgYHtyLCBtYWluLWZpZy1mZWN1bmRpdHktcGxvdHMsIGZpZy53aWR0aD04LjUsIGZpZy5oZWlnaHQ9MTB9CmN0bWF4X2ZlY3VuZF9wbG90ID0gY3RtYXhfcmVzaWRzICU+JSAgCiAgZHJvcF9uYShmZWN1bmRpdHkpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSByZXNpZHMsIHkgPSBmZWN1bmRpdHksIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuNSkgKyAKICBsYWJzKHggPSAiQ1RtYXggUmVzaWR1YWxzICjCsEMpIiwgCiAgICAgICB5ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmdnYXJyYW5nZShzaXplX2ZlY3VuZF9wbG90LCBjdG1heF9mZWN1bmRfcGxvdCwgbmNvbCA9IDEsIGNvbW1vbi5sZWdlbmQgPSBULCBsYWJlbHMgPSAiQVVUTyIsIGxlZ2VuZCA9ICJyaWdodCIpCmBgYAoKUHJldmlvdXMgc3R1ZGllcyBoYXZlIHNob3duIHRoYXQgdGhlIG1hZ25pdHVkZSBvZiB0aGUgc2l6ZS1mZWN1bmRpdHkgY29ycmVsYXRpb24gbWF5IGJlIGVudmlyb25tZW50YWxseS1kZXBlbmRlbnQuIFRoaXMgaXMgbm90IHZpc2libGUgaXMgdGhlIGRhdGEgZnJvbSB0aGVzZSBjb2xsZWN0aW9ucy4gCgpgYGB7ciBzdXBwLWZpZy1mZWMtc2l6ZS1jb3JyLXZzLXRlbXAsIGZpZy53aWR0aCA9IDQsIGZpZy5oZWlnaHQ9MTB9CmNvcnJfZGF0YSA9IGZ1bGxfZGF0YSAlPiUgCiAgZHJvcF9uYShmZWN1bmRpdHkpICU+JSAKICBmaWx0ZXIoc3BfbmFtZSAlaW4lIGMoIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMiLAogICAgICAgICAgICAgICAgICAgICAgICAiTGVwdG9kaWFwdG9tdXMgbWludXR1cyIsIAogICAgICAgICAgICAgICAgICAgICAgICAiU2tpc3RvZGlhcHRvbXVzIHNwIikpICU+JSAgCiAgZ3JvdXBfYnkoY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXAsIHNwX25hbWUpICU+JSAKICBzdW1tYXJpc2Uoc2l6ZV9mZWNfY29yciA9IGNvcihzaXplLCBmZWN1bmRpdHkpLAogICAgICAgICAgICBuID0gbigpLAogICAgICAgICAgICBtZWFuX2ZlY3VuZGl0eSA9IG1lYW4oZmVjdW5kaXR5KSkgJT4lIAogIGZpbHRlcihuID49IDMpICU+JSAKICB1bmdyb3VwKCkgJT4lICAKICBncm91cF9ieShzcF9uYW1lKSAlPiUgCiAgbXV0YXRlKHRlbXBfY2VudCA9IHNjYWxlKGNvbGxlY3Rpb25fdGVtcCwgc2NhbGUgPSBGKSkKCmdncGxvdChjb3JyX2RhdGEsIGFlcyh4ID0gdGVtcF9jZW50LCB5ID0gc2l6ZV9mZWNfY29yciwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4sIG5yb3cgPSAzKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDApICsKICBnZW9tX3BvaW50KHNpemUgPSAzKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGxpbmV3aWR0aCA9IDIpICsgCiAgc2NhbGVfY29sb3JfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiVGVtcGVyYXR1cmUgKGNlbnRlcmVkKSIsIAogICAgICAgeSA9ICJDb3JyZWxhdGlvbiBDb2VmZmljaWVudCIpICsgCiAgY29vcmRfY2FydGVzaWFuKHlsaW0gPSBjKC0xLCAxKSkgKwogIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCgojIGdncGxvdChjb3JyX2RhdGEsIGFlcyh4ID0gc2l6ZV9mZWNfY29ycikpICsKIyAgICAgZmFjZXRfd3JhcChzcF9uYW1lfi4sIG5yb3cgPSAzKSArCiMgICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDAuMikKYGBgCgoKIyMgT3RoZXIgcGF0dGVybnMgaW4gdmFyaWF0aW9uIAoKKkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMqIGlzIHRoZSBtb3N0IGFidW5kYW50IHNwZWNpZXMgZHVyaW5nIHRoZSB3aW50ZXIuIFRoZXJlIHdhcyBhIGxhcmdlIHNoaWZ0IGluIHRoZSBzaXplIG9mIG1hdHVyZSBmZW1hbGVzIHRvd2FyZHMgdGhlIGVuZCBvZiBEZWNlbWJlci4gVGhlc2UgbGFyZ2UgYW5kIHNtYWxsIGluZGl2aWR1YWxzIGFyZSB0aGUgc2FtZSBzcGVjaWVzIChjb25maXJtZWQgdmlhIENPSSBzZXF1ZW5jaW5nKSwgc3VnZ2VzdGluZyB0aGlzIHNoaWZ0IG1heSBpbnN0ZWFkIHJlZmxlY3QgYSB0cmFuc2l0aW9uIGZyb20gb25lIGdlbmVyYXRpb24gdG8gYW5vdGhlci4gVGhpcyBzaXplIGRpZmZlcmVuY2UgbWF5IGJlIGNhdXNlZCBieSBkaWZmZXJlbmNlcyBpbiB0aGUgZGV2ZWxvcG1lbnRhbCBlbnZpcm9ubWVudHMuIEZvciBleGFtcGxlLCBpbmRpdmlkdWFscyBkZXZlbG9waW5nIGluIEphbnVhcnkgZ3JvdyB1cCBhdCB2ZXJ5IGxvdyB0ZW1wZXJhdHVyZXMsIGFuZCB0aGVyZWZvcmUgbWF5IHJlYWNoIGxhcmdlciBzaXplcy4gVGhlc2UgaW5kaXZpZHVhbHMgb3Zlci1zdW1tZXIgaW4gZGVlcCB3YXRlcnMsIHRoZW4gcmUtZW1lcmdlIGluIE9jdG9iZXIgYW5kIHByb2R1Y2UgYSBuZXcgZ2VuZXJhdGlvbi4gV2F0ZXIgdGVtcGVyYXR1cmVzIGFyZSBzdGlsbCBmYWlybHkgaGlnaCB0aHJvdWdoIE5vdmVtYmVyLCB3aGljaCByZXN1bHRzIGluIGEgZ2VuZXJhdGlvbiBvZiBzbWFsbGVyIGluZGl2aWR1YWxzLiBUaGVzZSBpbmRpdmlkdWFscyBtYXR1cmUgaW4gdGltZSB0byBwcm9kdWNlIGEgbmV3IGdlbmVyYXRpb24gaW4gSmFudWFyeS4gCgpgYGB7ciBzdXBwLWZpZy1sc2ljLW1vcnBoLXNpemUsIGZpZy53aWR0aD05LCBmaWcuaGVpZ2h0PTl9CmZ1bGxfZGF0YSAlPiUgIAogIGZpbHRlcihzcF9uYW1lID09ICJMZXB0b2RpYXB0b211cyBzaWNpbGlzIikgJT4lIAogIGZpbHRlcihzZXggIT0gImp1dmVuaWxlIikgJT4lIAogIGdyb3VwX2J5KGNvbGxlY3Rpb25fZGF0ZSkgJT4lIAogIG11dGF0ZShzaXplX2NlbnRlciA9IHNjYWxlKHNpemUsIGNlbnRlciA9IFQsIHNjYWxlID0gRikpICU+JSAKICBnZ3Bsb3QoYWVzKHkgPSBmYWN0b3IoY29sbGVjdGlvbl9kYXRlKSwgeCA9IHNpemUsIGZpbGwgPSBjb2xsZWN0aW9uX3RlbXApKSArIAogIGZhY2V0X3dyYXAoc2V4fi4pICsgCiAgZ2VvbV9kZW5zaXR5X3JpZGdlcyhiYW5kd2lkdGggPSAwLjA0KSArIAogIGdlb21fdmxpbmUoeGludGVyY2VwdCA9IDAuODkpICsgCiAgbGFicyh4ID0gIlNpemUgKG1tKSIsCiAgICAgICB5ID0gIkRhdGUiLCAKICAgICAgIGZpbGwgPSAiQ29sbC4gVGVtcC4gKMKwQykiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIsCiAgICAgICAgYXhpcy50ZXh0LnkgPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyKSkKYGBgCgpBIHNpbWlsYXIsIGJ1dCBsZXNzIGRpc3RpbmN0IHBhdHRlcm4gY2FuIGJlIG9ic2VydmVkIGluIEwuIG1pbnV0dXMgaW5kaXZpZHVhbHMgYXMgd2VsbC4gCmBgYHtyIHN1cHAtZmlnLWxtaW4tbW9ycGgtc2l6ZSwgZmlnLndpZHRoPTksIGZpZy5oZWlnaHQ9OX0KZnVsbF9kYXRhICU+JSAgCiAgZmlsdGVyKHNwX25hbWUgPT0gIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMiKSAlPiUgCiAgZmlsdGVyKHNleCAhPSAianV2ZW5pbGUiKSAlPiUgCiAgZ2dwbG90KGFlcyh5ID0gZmFjdG9yKGNvbGxlY3Rpb25fZGF0ZSksIHggPSBzaXplLCBmaWxsID0gY29sbGVjdGlvbl90ZW1wKSkgKyAKICBmYWNldF93cmFwKHNleH4uKSArIAogIGdlb21fZGVuc2l0eV9yaWRnZXMoYmFuZHdpZHRoID0gMC4wNCkgKyAKICBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSAwLjY5KSArIAogIGxhYnMoeCA9ICJTaXplIChtbSkiLAogICAgICAgeSA9ICJEYXRlIiwgCiAgICAgICBmaWxsID0gIkNvbGwuIFRlbXAuICjCsEMpIikgKyAKICBjb29yZF9jYXJ0ZXNpYW4oeGxpbSA9IGMoMC41LDAuOSkpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IiwKICAgICAgICBheGlzLnRleHQueSA9IGVsZW1lbnRfdGV4dChzaXplID0gMTIpKQpgYGAKCiMjIERpc3RyaWJ1dGlvbiBMYWcgTm9uLUxpbmVhciBNb2RlbCAoRExOTSBhcHByb2FjaCkgCgpEaXN0cmlidXRlZCBsYWcgbW9kZWxzIGV4YW1pbmUgYSByZXNwb25zZSB2YXJpYWJsZSwgbWVhc3VyZWQgYXQgbXVsdGlwbGUgdGltZSBwb2ludHMsIGFzIGEgZnVuY3Rpb24gb2YgdGhlIGxhZ2dlZCBvY2N1cnJlbmNlIG9mIHNvbWUgcHJlZGljdG9yIHZhcmlhYmxlIChyZXNwb25zZSB5IGF0IHRpbWUgdCBhcyBhIGZ1bmN0aW9uIG9mIHByZWRpY3RvciB4KHQtbGFnKS4gVGhpcyBtZXRob2QgdXRpbGl6ZXMgYSBiaS1kaW1lbnNpb25hbCBkb3NlLWxhZy1yZXNwb25zZSBmdW5jdGlvbiwgd2hpY2ggZXNzZW50aWFsbHkgZXhhbWluZXMgbm90IG9ubHkgdGhlIGRvc2UgZWZmZWN0LCBidXQgdGhlIGVmZmVjdCBvZiB0aGUgdGltaW5nIG9mIHRoZSBkb3NlLiAKCgpgYGB7ciBjb21waWxpbmdfdGVtcF9sYWdzfQojIFJ1biB0aGlzIGNvZGUsIHNhdmUgdGhlIHByb2R1Y3QsIGFuZCB0aGVuIGp1c3QgcmVhZCBpbiB0aGUgdGVtcCBsYWcgZGF0YSBvYmplY3QuIFRha2VzIHRvbyBsb25nIHRvIHJ1biBlYWNoIHRpbWUgdGhpcyBkb2N1bWVudCBpcyBrbml0LiAKCiMgbGFnX3RlbXBzID0gdGVtcF9kYXRhICU+JQojICAgZ3JvdXBfYnkoZGF0ZSwgaG91cikgJT4lCiMgICBzdW1tYXJpemUoIm1lYW5fdGVtcCIgPSBtZWFuKHRlbXAsIG5hLnJtID0gVCkpICU+JQojICAgdW5ncm91cCgpICU+JQojICAgbXV0YXRlKHBvaW50X251bSA9IHJvd19udW1iZXIoKSkKIyAKIyB1bmlxX2RheXMgPSBsZW5ndGgodW5pcXVlKGxhZ190ZW1wcyRkYXRlKSkKIyAKIyBnID0gZ2FtKG1lYW5fdGVtcCB+IHMocG9pbnRfbnVtLCBicz0iY3IiLCBrPXVuaXFfZGF5cyArIDEwKSwKIyAgICAgbWV0aG9kID0gIlJFTUwiLAojICAgICBkYXRhID0gbGFnX3RlbXBzKQojIAojIHBvaW50cyA9IHNlcSgxLCBucm93KGxhZ190ZW1wcyksIGxlbmd0aC5vdXQgPSBsZW5ndGgobGFnX3RlbXBzJGhvdXIpKQojIAojIGRmLnJlcyA9IGRmLnJlc2lkdWFsKGcpCiMgCiMgcHJlZF90ZW1wcyA9IHByZWRpY3QoZywgbmV3ZGF0YSA9IGxhZ190ZW1wcywgdHlwZSA9ICJyZXNwb25zZSIsIHNlLmZpdCA9IFRSVUUpCiMgCiMgbGFnX3RlbXBzID0gbGFnX3RlbXBzICU+JQojICAgbXV0YXRlKHRyZW5kX1QgPSBwcmVkX3RlbXBzJGZpdCwKIyAgICAgICAgICB0cmVuZF9zZSA9IHByZWRfdGVtcHMkc2UuZml0LAojICAgICAgICAgIHRlbXBfZGlmZiA9IG1lYW5fdGVtcCAtIHRyZW5kX1QpCiMgCiMgd3JpdGUuY3N2KGxhZ190ZW1wcywgZmlsZSA9ICIuL091dHB1dC9EYXRhL2xhZ190ZW1wcy5jc3YiLCByb3cubmFtZXMgPSBGKQoKYGBgCgpgYGB7ciBzdXBwLWZpZy1kYWlseS1tYXgtY3RtYXgsIGZpZy53aWR0aD05LCBmaWcuaGVpZ2h0PTR9IAoKZGxubV9kYXRhID0gZnVsbF9kYXRhICU+JSAgCiAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogIGZpbHRlcihzcF9uYW1lICVpbiUgYygKICAgICJMZXB0b2RpYXB0b211cyBzaWNpbGlzIiwKICAgICJMZXB0b2RpYXB0b211cyBtaW51dHVzIgogICkpICU+JSAKICBzZWxlY3QoY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXAsIHNwX25hbWUsIGN0bWF4KSAlPiUgCiAgZ3JvdXBfYnkoY29sbGVjdGlvbl9kYXRlLCBjb2xsZWN0aW9uX3RlbXAsIHNwX25hbWUpICU+JSAgCiAgc3VtbWFyaXNlKG1lYW5fY3RtYXggPSBtZWFuKGN0bWF4LCBuYS5ybSA9IFQpLAogICAgICAgICAgICBzYW1wbGUgPSBuKCkpCgp0ZW1wX2RhdGEgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpc2UobWVhbl90ZW1wID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgbWF4X3RlbXAgPSBtYXgodGVtcCwgbmEucm0gPSBUKSkgJT4lIAogIHJpZ2h0X2pvaW4oZGxubV9kYXRhLCBieSA9IGpvaW5fYnkoImRhdGUiID09ICJjb2xsZWN0aW9uX2RhdGUiKSkgJT4lIAogIGdncGxvdChhZXMoeCA9IG1heF90ZW1wLCB5ID0gbWVhbl9jdG1heCkpICsgCiAgZmFjZXRfd3JhcCgufnNwX25hbWUpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImdhbSIpICsgCiAgZ2VvbV9wb2ludCgpICsgCiAgbGFicyh4ID0gIk1heCBEYWlseSBUZW1wLiAowrBDKSIsCiAgICAgICB5ID0gIk1lYW4gQ1RtYXggKMKwQykiKSArIAogIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKICB0aGVtZShzdHJpcC50ZXh0LnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IDEyKSkKYGBgCgpgYGB7ciBzdXBwLWZpZy1kbG5tLXBsb3R9CgpzcF9saXN0ID0gdW5pcXVlKGRsbm1fZGF0YSRzcF9uYW1lKQoKZm9yKGxhZ19zcGVjaWVzIGluIHNwX2xpc3QpewogIAogIGRsbm1fZGF0YV9zcCA9IGRsbm1fZGF0YSAlPiUgCiAgICBmaWx0ZXIoc3BfbmFtZSA9PSBsYWdfc3BlY2llcykKICAKICAjIFdlIG5lZWQgdG8gZXN0aW1hdGUgYSBtYXRyaXggb2YgZXhwb3N1cmUgaGlzdG9yaWVzIGZvciBlYWNoIG9ic2VydmF0aW9uLiBUaGlzIGNvbnRhaW5zIHRoZSBzZXJpZXMgb2YgZXhwb3N1cmVzIGF0IGVhY2ggbGFnIChsKSBmb3IgZWFjaCBvZiB0aGUgbiBvYnNlcnZhdGlvbnMsIGNvbnN0cmFpbmVkIGJldHdlZW4gbDAgKG1pbmltdW0gbGFnKSBhbmQgTCAobWF4IGxhZykuIAogIAogIGRhdGVzID0gZGxubV9kYXRhX3NwJGNvbGxlY3Rpb25fZGF0ZSAjIEZvciBlYWNoIG9mIHRoZXNlIGRhdGVzLCBtYWtlIGEgdmVjdG9yIG9mIHRoZSBwYXN0IDMwIGRheXMgKGluY2x1ZGluZyB0aGUgZGF5IG9mIGNvbGxlY3Rpb24pLiBOT1RFOiBEb24ndCB1c2UgJ3VuaXF1ZScgZGF0ZXMgaGVyZSBzaW5jZSBzb21lIGNvbGxlY3Rpb25zIGhhZCBtdWx0aXBsZSBzcGVjaWVzCiAgCiAgZXhwX2hpc3RfeiA9IGRhdGEuZnJhbWUoKQogIGV4cF9oaXN0X3RyZW5kID0gZGF0YS5mcmFtZSgpCiAgCiAgZm9yKGQgaW4gZGF0ZXMpewogICAgCiAgICBoaXN0b3J5ID0gbGFnX3RlbXBzICU+JSAKICAgICAgZmlsdGVyKGRhdGUgPD0gZCAmIGRhdGUgPiBkIC0gMTApICU+JSAKICAgICAgYXJyYW5nZShkZXNjKGRhdGUpLCBkZXNjKGhvdXIpKSAlPiUgCiAgICAgIG11dGF0ZShsYWcgPSByb3dfbnVtYmVyKCkgLSAxKSAlPiUgCiAgICAgIHNlbGVjdChsYWcsIG1lYW5fdGVtcCwgdGVtcF9kaWZmKQogICAgCiAgICB6X3ZlYyA9IHNjYWxlKGhpc3RvcnkkbWVhbl90ZW1wKVssMV0KICAgIG5hbWVzKHpfdmVjKSA9IGhpc3RvcnkkbGFnCiAgICAKICAgIHRyZW5kX3ZlYyA9IGhpc3RvcnkkdGVtcF9kaWZmCiAgICBuYW1lcyh0cmVuZF92ZWMpID0gaGlzdG9yeSRsYWcKICAgIAogICAgZXhwX2hpc3RfeiA9IGJpbmRfcm93cyhleHBfaGlzdF96LCB6X3ZlYykKICAgIGV4cF9oaXN0X3RyZW5kID0gYmluZF9yb3dzKGV4cF9oaXN0X3RyZW5kLCB0cmVuZF92ZWMpCiAgICAKICB9CiAgCiAgI3ByaW50KG1heChleHBfaGlzdF90cmVuZCwgbmEucm0gPSBUKSkKICAKICAjIFRoZSBjcm9zcy1iYXNpcyBmdW5jdGlvbiBmcm9tIGRsbm0gd2lsbCB1c2UgdGhlIGNsYXNzIG9mIHRoZSB4IHBhcmFtZXRlciB0byBkZXRlcm1pbmUgd2hhdCB0byBkby4gSW4gb3VyIGNhc2UsIHdlIG5lZWQgdG8gcHJvdmlkZSBpdCB3aXRoIHRoZSBtYXRyaXggb2YgZXhwb3N1cmUgaGlzdG9yaWVzIGZvciByZWFjaCBvYnNlcnZhdGlvbiAocm93KSBhbmQgbGFnIChjb2x1bW4pLiAKICAKICBjYl90ZW1wcyA9IGNyb3NzYmFzaXMoZXhwX2hpc3RfdHJlbmQsIGxhZyA9IGMoMCxkaW0oZXhwX2hpc3RfdHJlbmQpWzJdLTEpLCAKICAgICAgICAgICAgICAgICAgICAgICAgYXJndmFyID1saXN0KGZ1bj0iY3IiLGRmPTMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgYXJnbGFnPWxpc3QoZnVuPSJjciIsZGY9MyxpbnRlcmNlcHQ9VCkpCiAgCiAgI3N1bW1hcnkoY2JfdGVtcHMpCiAgCiAgcGVuYWxpemVkX21hdCA8LSBjYlBlbihjYl90ZW1wcykKICAKICAjZml0dGluZyBHQU0KICBsYWcuZ2FtID0gZ2FtKGRhdGEgPSBkbG5tX2RhdGFfc3AsIAogICAgICAgICAgICAgICAgbWVhbl9jdG1heCB+IGNvbGxlY3Rpb25fdGVtcCArIGNiX3RlbXBzLCAKICAgICAgICAgICAgICAgIG1ldGhvZCA9ICJHQ1YuQ3AiLAogICAgICAgICAgICAgICAgcGFyYVBlbj1saXN0KGNiX3RlbXBzPXBlbmFsaXplZF9tYXQpKQogIAogICMgc3VtbWFyeShsYWcuZ2FtKQogICMgQUlDKGxhZy5nYW0pCiAgCiAgI2VzdGltYXRpb24gb2YgZXhwb3N1cmVzIGVmZmVjdHMKICAKICAjZGVmYXVsdCBwbG90cwogIHByZWRfZ2FtX1pzPC1jcm9zc3ByZWQoY2JfdGVtcHMsIGxhZy5nYW0sIAogICAgICAgICAgICAgICAgICAgICAgICAgY3VtdWw9RiwgY2VuPTAsIGNpLmxldmVsID0gMC45NSwKICAgICAgICAgICAgICAgICAgICAgICAgIGF0PXNlcSgtNCw0LCAwLjEpKQogIAogIHBsb3QocHJlZF9nYW1fWnMsICJjb250b3VyIiwgbWFpbiA9IGxhZ19zcGVjaWVzLCAKICAgICAgICAgICAgICB4bGFiID0gIlRlbXBlcmF0dXJlIERldmlhdGlvbiAowrBDKSIsIAogICAgICAgICAgICAgIHlsYWIgPSAiSG91cnMgYmVmb3JlIGNvbGxlY3Rpb24iKQogIAogIAogICMgCiAgIyBwbG90KHByZWRfZ2FtX1pzLCBib3JkZXIgPSAyLCBjdW11bD1GLAogICMgICAgICAgdGhldGE9MTEwLHBoaT0yMCxsdGhldGE9LTgwKQogIAogICMgcGxvdChwcmVkX2dhbV9acywgInNsaWNlcyIsCiAgIyAgICAgIHZhciA9IGMoMywtMyksCiAgIyAgICAgIGxhZyA9IGMoMSwyMDApLAogICMgICAgICBjb2wgPSAyKQogICMgCn0KCgpgYGAKCgojIyBNaXNjZWxsYW55CgoKYGBge3Igc3VwcC1maWctcmFtcC1yYXRlLCBmaWcud2lkdGggPSA3LCBmaWcuaGVpZ2h0PTV9CnJ1bl9zdGFydHMgPSB0ZW1wX3JlY29yZCAlPiUgCiAgZ3JvdXBfYnkocnVuKSAlPiUgCiAgZmlsdGVyKG1pbnV0ZV9wYXNzZWQgPD0gMykgJT4lIAogIHN1bW1hcmlzZShzdGFydF90ZW1wID0gbWVhbih0ZW1wX0MpKSAlPiUgCiAgbXV0YXRlKCJ0ZW1wX2JpbiIgPSBjdXRfbnVtYmVyKHN0YXJ0X3RlbXAsIDQpLAogICAgICAgICB0ZW1wX2JpbiA9IGNhc2Vfd2hlbigKICAgICAgICAgICB0ZW1wX2JpbiA9PSAiWzIuNyw5LjI2XSIgfiAiWzIuN8KwQyAtIDkuMjbCsENdIiwKICAgICAgICAgICB0ZW1wX2JpbiA9PSAiKDkuMjYsMTMuOV0iIH4gIls5LjI2wrBDIC0gMTMuOcKwQ10iLAogICAgICAgICAgIHRlbXBfYmluID09ICIoMTMuOSwyMV0iIH4gIlsxMy45wrBDIC0gMjHCsENdIiwKICAgICAgICAgICB0ZW1wX2JpbiA9PSAiKDIxLDMwLjVdIiB+ICJbMjHCsEMgLSAzMC41wrBDXSIKICAgICAgICAgKSkKCnJhbXBfcmVjb3JkMiA9IHJhbXBfcmVjb3JkICU+JSAKICBncm91cF9ieShydW4sIG1pbnV0ZV9pbnRlcnZhbCkgJT4lIAogIHN1bW1hcmlzZShtZWFuX3JhbXAgPSBtZWFuKHJhbXBfcGVyX21pbnV0ZSkpICU+JSAKICB1bmdyb3VwKCkgJT4lIAogIGxlZnRfam9pbihydW5fc3RhcnRzKSAlPiUgCiAgbXV0YXRlKHRlbXBfYmluID0gZmN0X3Jlb3JkZXIodGVtcF9iaW4sIHN0YXJ0X3RlbXAsIC5mdW4gPSBtZWFuKSkKCmdncGxvdChyYW1wX3JlY29yZDIsIGFlcyh4ID0gbWludXRlX2ludGVydmFsLCB5ID0gbWVhbl9yYW1wKSkgKyAKICBmYWNldF93cmFwKHRlbXBfYmlufi4pICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMC4zLCBjb2xvdXIgPSAiZ3JleSIpICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMC4xLCBjb2xvdXIgPSAiZ3JleSIpICsgCiAgZ2VvbV9oZXgoYWVzKGZpbGwgPSBjdXQoLi5jb3VudC4uLCBjKDIsIDUsIDEwLCAyMCwgMzAsIDQwLCA1MCkpKSwKICAgICAgICAgICBiaW5zID0gMzApICsgCiAgc2NhbGVfZmlsbF92aXJpZGlzX2QobmFtZT0iTnVtYmVyIG9mIE9ic2VydmF0aW9ucyIsCiAgICAgICAgICAgICAgICAgICAgICAgbGFiZWxzID0gYygiPDUiLCAiNS05IiwgIjEwLTE5IiwgIjIwLTI5IiwgIjMwLTM5IiwgIjQwLTUwIiksCiAgICAgICAgICAgICAgICAgICAgICAgb3B0aW9uID0gIm1ha28iKSArIAogIGxhYnMoeSA9ICJSYW1wIFJhdGUgKGRlZy4gQyAvIG1pbi4pIiwKICAgICAgIHggPSAiVGltZSBpbnRvIHJ1biAobWludXRlKSIpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoYmFzZV9zaXplID0gMTIpCmBgYAoKCgoKCgpgYGB7cn0KCmlmKHByZWRpY3RfdnVsbiA9PSBGKXsKCiAga25pdHI6OmtuaXRfZXhpdCgpCgp9CgpgYGAKCiMjIEhpbmRjYXN0aW5nIHZ1bG5lcmFiaWxpdHkgd2l0aCBkaWZmZXJlbnQgYWNjbGltYXRpb24gc2NlbmFyaW9zClVzaW5nIHRoZSBvYnNlcnZlZCB0aGVybWFsIGxpbWl0IGRhdGEsIHdlIGNhbiBwcm9kdWNlIGEgaGluZGNhc3Qgb2YgdGhlcm1hbCBzdHJlc3MgZm9yIExha2UgQ2hhbXBsYWluIGNvcGVwb2RzLiBGb3IgdGhlc2UgaW5pdGlhbCBhc3NheXMsIHdlIHdpbGwgZGVmaW5lIHRoZXJtYWwgc3RyZXNzIGFzIGFueSB0aW1lIHdoZW4gbWF4aW11bSBkYWlseSB3YXRlciB0ZW1wZXJhdHVyZSBpcyB3aXRoaW4gNcKwQyBvZiBjb3BlcG9kIENUbWF4IG9yIGhpZ2hlciAoaS5lLiB3YXJtaW5nIHRvbGVyYW5jZSBpcyBsZXNzIHRoYW4gNcKwQykuIFdlIHdpbGwgdXNlIHRocmVlIGRpZmZlcmVudCBzY2VuYXJpb3M6IDEpIGVhY2ggc3BlY2llcyBoYXMgYSB1bmlxdWUgYnV0IGZpeGVkIHRoZXJtYWwgbGltaXQgKGF2ZXJhZ2UgbWVhc3VyZWQgQ1RtYXgpLCAyKSBzcGVjaWVzIGhhdmUgdW5pcXVlIHRoZXJtYWwgbGltaXRzIGFuZCBhcmUgYWJsZSB0byBhY2NsaW1hdGUgKGNvbnN0YW50IEFSUiBhY3Jvc3MgYWxsIHNwZWNpZXMgdXNlZCB0byBwcmVkaWN0IENUbWF4IGJhc2VkIG9uIGF2ZXJhZ2UgZGFpbHkgdGVtcGVyYXR1cmVzKSwgYW5kIDMpIHNwZWNpZXMgaGF2ZSB1bmlxdWUgdGhlcm1hbCBsaW1pdHMgYW5kIHNwZWNpZXMtc3BlY2lmaWMgYWNjbGltYXRpb24gKENUbWF4IHByZWRpY3RlZCB1c2luZyB3aGljaGV2ZXIgZW52aXJvbm1lbnRhbCBmYWN0b3IgYW5kIGR1cmF0aW9uIGlzIHRoZSBzdHJvbmdlc3QgY2FuZGlkYXRlIGZvciBkcml2aW5nIGFjY2xpbWF0aW9uIC0gZnJvbSB0aGUgY29ycmVsYXRpb24gYW5hbHlzaXMpLiBJbiBhbGwgY2FzZXMsIGRhdGEgaXMgZmlsdGVyZWQgdG8ganVzdCB0aGVybWFsIGxpbWl0cyBvZiBhZHVsdCBmZW1hbGVzLiAKCiMjIyBTY2VuYXJpbyAxCmBgYHtyfQptZWFuX2N0bWF4ID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JSAKICBzdW1tYXJpemUoIm1lYW5fY3RtYXgiID0gbWVhbihjdG1heCkpICU+JSAKICBhcnJhbmdlKG1lYW5fY3RtYXgpCgprbml0cjo6a2FibGUobWVhbl9jdG1heCkKYGBgCgpgYGB7cn0KIyAjIENvbnN0cnVjdHMgdGhlIFVSTCBmb3IgdGhlIGZ1bGwgdGVtcGVyYXR1cmUgZGF0YSBzZXQ7IFJVTiBUSElTIE9OQ0UKIyBoaW5kX3VybCA9IGNvbnN0cnVjdE5XSVNVUkwoc2l0ZU51bWJlcnMgPSBzaXRlTnVtYmVyLCBwYXJhbWV0ZXJDZCA9IHBhcmFtZXRlckNkLCBzZXJ2aWNlID0gInV2IikKIyAKIyBoaW5kX3RlbXBfZGF0YSA9IGltcG9ydFdhdGVyTUwxKGhpbmRfdXJsLCBhc0RhdGVUaW1lID0gVCkgJT4lCiMgICBtdXRhdGUoImRhdGUiID0gYXMuRGF0ZShkYXRlVGltZSkpICU+JQojICAgc2VsZWN0KGRhdGUsICJ0ZW1wIiA9IFhfMDAwMTBfMDAwMDApCiMgCiMgd3JpdGUudGFibGUoeCA9IGhpbmRfdGVtcF9kYXRhLCBmaWxlID0gImhpbmRjYXN0X3RlbXBzLmNzdiIsIHJvdy5uYW1lcyA9IEYsIHNlcCA9ICIsIikKYGBgCgpgYGB7cn0KIyBnZ3Bsb3QoaGluZF90ZW1wX2RhdGEsIGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXApKSArIAojICAgZ2VvbV9saW5lKGxpbmV3aWR0aCA9IDAuMSkgKyAKIyAgIGxhYnMoeCA9ICJEYXRlIiwgCiMgICAgICAgIHkgPSAiV2F0ZXIgVGVtcGVyYXR1cmUgKMKwQykiKSArCiMgICB0aGVtZV9tYXR0KCkKYGBgCgpJbiB0aGUgc2ltcGxlc3Qgc2NlbmFyaW8sIHNwZWNpZXMgdGhlcm1hbCBsaW1pdHMgYXJlIHN0YXRpYyB0aHJvdWdoIHRpbWUsIHJlcHJlc2VudGVkIGJ5IHRoZSBhdmVyYWdlIENUbWF4IG9mIGFkdWx0IGZlbWFsZSBjb3BlcG9kcy4gSW4gdGhpcyBzY2VuYXJpbywgb25seSB0aHJlZSBvZiB0aGUgc2V2ZW4gb2JzZXJ2ZWQgc3BlY2llcyBhcmUgZXhwb3NlZCB0byB0aGVybWFsIHN0cmVzcyAodGVtcGVyYXR1cmVzIHdpdGhpbiA1wrBDIG9mIENUbWF4KS4gVGVtcGVyYXR1cmVzIGFwcHJvYWNoZWQgdGhlIHRoZXJtYWwgbGltaXQgb2YgKkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMqIG9uIGEgaGFuZGZ1bCBvZiBkYXlzLiBCeSBjb250cmFzdCwgKlNlbmVjZWxsYSBjYWxhbm9pZGVzKiBhbmQgKkxpbW5vY2FsYW51cyBtYWNydXJ1cyogd2VyZSBib3RoIGV4cG9zZWQgdG8gc3Vic3RhbnRpYWwgdGhlcm1hbCBzdHJlc3MgdGhyb3VnaG91dCBhIGxhcmdlIHBvcnRpb24gb2YgdGhlIHllYXIsIGxpa2VseSBleHBsYWluaW5nIHdoeSB0aGVzZSBzcGVjaWVzIGFyZSBhYnNlbnQgZnJvbSB0aGUgY29tbXVuaXR5IGZvciB0aGUgc3VtbWVyIGFuZCBmYWxsIHBlcmlvZHMuIAoKYGBge3Igc3VwcC1maWctaGluZDFfc2NlbmFyaW8sIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD01fQpoaW5kMV9kYXRhID0gaGluZF90ZW1wX2RhdGEgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpemUoImRhaWx5X21heCIgPSBtYXgodGVtcCksCiAgICAgICAgICAgICJkYWlseV9tZWFuIiA9IG1lYW4odGVtcCksKSAlPiUgCiAgYmluZF9jb2xzKHBpdm90X3dpZGVyKG1lYW5fY3RtYXgsIG5hbWVzX2Zyb20gPSBzcF9uYW1lLCB2YWx1ZXNfZnJvbSA9IG1lYW5fY3RtYXgpKSAlPiUgIAogIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSwgLWRhaWx5X21heCwgLWRhaWx5X21lYW4pLAogICAgICAgICAgICAgICBuYW1lc190byA9ICJzcGVjaWVzIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJtZWFuX2N0bWF4IikgJT4lICAKICBtdXRhdGUobGltX2RpZmYgPSBtZWFuX2N0bWF4IC0gZGFpbHlfbWF4KSAlPiUgIAogIG11dGF0ZShkb3kgPSB5ZGF5KGRhdGUpLAogICAgICAgICAibWV0aG9kIiA9ICJOb19hY2NsaW1hdGlvbiIpCgpoaW5kX2RhaWx5X3RlbXBfZGF0YSA9IGhpbmRfdGVtcF9kYXRhICU+JQogIHVuZ3JvdXAoKSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcmlzZShtZWFuX3RlbXAgPSBtZWFuKHRlbXApLAogICAgICAgICAgICBtZWRfdGVtcCA9IG1lZGlhbih0ZW1wKSwKICAgICAgICAgICAgdmFyX3RlbXAgPSB2YXIodGVtcCksIAogICAgICAgICAgICBtaW5fdGVtcCA9IG1pbih0ZW1wKSwgCiAgICAgICAgICAgIG1heF90ZW1wID0gbWF4KHRlbXApKSAlPiUgCiAgbXV0YXRlKCJyYW5nZV90ZW1wIiA9IG1heF90ZW1wIC0gbWluX3RlbXApCgojdGFibGUoaGluZDFfZGF0YSRzcGVjaWVzKQoKaGluZDFfZGF0YSAlPiUgCiAgZmlsdGVyKGxpbV9kaWZmIDw9IDUpICU+JSAgCiAgZ2dwbG90KGFlcyh4ID0gZG95LCB5ID0gbGltX2RpZmYsIGNvbG91ciA9IHNwZWNpZXMpKSArCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMCkgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSA1LCAKICAgICAgICAgICAgIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC41KSArCiAgZ2VvbV9zbW9vdGgoc2UgPSBGKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgWWVhciIsIAogICAgICAgeSA9ICJQcmVkaWN0ZWQgV2FybWluZyBUb2xlcmFuY2UgXG4owrBDIEFib3ZlIERhaWx5IE1heCkiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKIyMjIFNjZW5hcmlvIDIKSW4gdGhlIHNlY29uZCBzY2VuYXJpbywgdGhlcm1hbCBsaW1pdHMgdmFyeSB3aXRoaW4gYW5kIGJldHdlZW4gc3BlY2llcy4gQSBzaW1wbGUgbW9kZWwgaXMgdXNlZCB0byBwcmVkaWN0IHNwZWNpZXMgdGhlcm1hbCBsaW1pdHMgYmFzZWQgb24gbWVhbiBkYWlseSB0ZW1wZXJhdHVyZSAoQ1RtYXggYXMgYSBmdW5jdGlvbiBvZiBzcGVjaWVzIGFuZCBjb2xsZWN0aW9uIHRlbXBlcmF0dXJlLCBidXQgd2l0aG91dCB0aGUgaW50ZXJhY3Rpb24gYmV0d2VlbiB0aGVzZSB0d28gZmFjdG9ycykuIFRoZXNlIHByZWRpY3RlZCB0aGVybWFsIGxpbWl0cyBhcmUgdGhlbiBjb21wYXJlZCBhZ2FpbnN0IHRoZSBtYXhpbXVtIGRhaWx5IHRlbXBlcmF0dXJlIHRvIGVzdGltYXRlIHRoZXJtYWwgc3RyZXNzLCBhcyBpbiBTY2VuYXJpbyAxLiBJbmNsdWRpbmcgdGhpcyBzaW1wbGUgZm9ybSBvZiBhY2NsaW1hdGlvbiBpbiB0aGUgbW9kZWwgcmVkdWNlZCB0aGUgZGVncmVlIG9mIHRoZXJtYWwgc3RyZXNzIGZvciBlYWNoIHNwZWNpZXMsIGVsaW1pbmF0aW5nIGl0IGVudGlyZWx5IGZvciAqTGVwdG9kaWFwdG9tdXMgc2ljaWxpcyouIE5vdGUgdGhhdCB0aGUgbWFnbml0dWRlIG9mIHRoZSBwcmVkaWN0ZWQgc3RyZXNzIGlzICBsb3cgZW5vdWdoIHRoYXQgcmVtb3ZpbmcgdGhlIDXCsEMgYnVmZmVyIGFyb3VuZCB0aGUgcHJlZGljdGVkIHRoZXJtYWwgbGltaXRzIHdvdWxkIGFjdHVhbGx5IGxpbWl0IHByZWRpY3RlZCB0aGVybWFsIHN0cmVzcyB0byBqdXN0IGEgZmV3IGRheXMgZm9yICpTZW5lY2VsbGEgY2FsYW5vaWRlcyouIAoKYGBge3Igc3VwcC1maWctaGluZDJfc2NlbmFyaW8sIGZpZy5oZWlnaHQ9NSwgZmlnLndpZHRoPTEwfQpoaW5kY2FzdF9tb2RlbDEgPSBsbShkYXRhID0gZmlsdGVyKGZ1bGxfZGF0YSwgc2V4ID09ICJmZW1hbGUiKSwKICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiBjb2xsZWN0aW9uX3RlbXAgKyBzcF9uYW1lKQoKaGluZDJfZGF0YSA9IGhpbmRfdGVtcF9kYXRhICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXplKCJjb2xsZWN0aW9uX3RlbXAiID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgImRhaWx5X21heCIgPSBtYXgodGVtcCkpICU+JSAKICBiaW5kX2NvbHMoCiAgICBwaXZvdF93aWRlcihtZWFuX2N0bWF4LCAKICAgICAgICAgICAgICAgIG5hbWVzX2Zyb20gPSBzcF9uYW1lLCAKICAgICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbWVhbl9jdG1heCkpICU+JSAKICBwaXZvdF9sb25nZXIoY29scyA9IGMoLWRhdGUsIC1kYWlseV9tYXgsIC1jb2xsZWN0aW9uX3RlbXApLAogICAgICAgICAgICAgICBuYW1lc190byA9ICJzcF9uYW1lIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJtZWFuX2N0bWF4IikgJT4lIAogIHNlbGVjdCgtbWVhbl9jdG1heCkgJT4lIAogIG11dGF0ZSgicHJlZF9jdG1heCIgPSBwcmVkaWN0LmxtIChoaW5kY2FzdF9tb2RlbDEsIG5ld2RhdGEgPSAuKSkgJT4lIAogIHNlbGVjdChkYXRlLCAiZGFpbHlfbWVhbiIgPSBjb2xsZWN0aW9uX3RlbXAsIGRhaWx5X21heCwgInNwZWNpZXMiID0gc3BfbmFtZSwgcHJlZF9jdG1heCkgJT4lIAogIG11dGF0ZShsaW1fZGlmZiA9IHByZWRfY3RtYXggLSBkYWlseV9tYXgpICU+JSAKICAjZmlsdGVyKGxpbV9kaWZmIDw9IDApICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoZGF0ZSksCiAgICAgICAgICJtZXRob2QiID0gIkNvbnN0YW50X2FjY2xpbWF0aW9uIikKCiMgZ2dwbG90KGhpbmQyX2RhdGEsIGFlcyh4ID0gZGFpbHlfbWVhbiwgeSA9IHByZWRfY3RtYXgsIGNvbG91ciA9IHNwZWNpZXMpKSArCiMgICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iKSAKCiMgdGFibGUoaGluZDJfZGF0YSRzcGVjaWVzKQpoaW5kMl9kYXRhICU+JSAgCiAgZmlsdGVyKGxpbV9kaWZmIDw9IDUpICU+JSAgCiAgZ2dwbG90KGFlcyh4ID0gZG95LCB5ID0gbGltX2RpZmYsIGNvbG91ciA9IHNwZWNpZXMpKSArCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMCkgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSA1LCAKICAgICAgICAgICAgIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC41KSArCiAgZ2VvbV9zbW9vdGgoKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgWWVhciIsIAogICAgICAgeSA9ICJQcmVkaWN0ZWQgV2FybWluZyBUb2xlcmFuY2UgXG4owrBDIEFib3ZlIERhaWx5IE1heCkiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKIyMjIFNjZW5hcmlvIDMKVGhlIGZpbmFsIHNjZW5hcmlvIGFsbG93cyB0aGUgZW52aXJvbm1lbnRhbCB2YXJpYWJsZSB1c2VkIHRvIHByZWRpY3QgQ1RtYXggdG8gdmFyeSBiZXR3ZWVuIHNwZWNpZXMuIEZvciBzcGVjaWVzIG9ic2VydmVkIGluIGZld2VyIHRoYW4gNSBjb2xsZWN0aW9ucywgd2UgdXNlIHRoZSBzYW1lIGFwcHJvYWNoIGFzIGluIFNjZW5hcmlvIDIuIEZvciBzcGVjaWVzIG9ic2VydmVkIGluIG1vcmUgdGhhbiA1IGNvbGxlY3Rpb25zLCBob3dldmVyLCB0aGUgZmFjdG9yIHdpdGggdGhlIHN0cm9uZ2VzdCBjb3JyZWxhdGlvbiB3aXRoIENUbWF4IGlzIHVzZWQgdG8gcHJlZGljdCB0aGVybWFsIGxpbWl0cy4gVGhlc2UgZmFjdG9ycyBhcmUgaW5jbHVkZWQgYmVsb3cuCgpgYGB7cn0KaGluZF9wcmVkcyA9IGNvcnJfdmFscyAlPiUgIAogIGZpbHRlcihzaWcgPT0gIlNpZy4iKSAlPiUgCiAgZHJvcF9uYShjb3JyZWxhdGlvbikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JQogIGFycmFuZ2UoZGVzYyhjb3JyZWxhdGlvbikpICU+JSAKICBzbGljZV9oZWFkKG4gPSAxKSAlPiUgCiAgc2VsZWN0KCJTcGVjaWVzIiA9IHNwX25hbWUsICJQcmVkaWN0b3IiID0gcGFyYW1ldGVyLCAiRHVyYXRpb24iID0gZHVyYXRpb24sICJDb3JyZWxhdGlvbiIgPSBjb3JyZWxhdGlvbiwgIlAtVmFsdWUiID0gcC52YWx1ZSkKCmtuaXRyOjprYWJsZShoaW5kX3ByZWRzLCBhbGlnbiA9ICJjIikKYGBgCgpgYGB7cn0KaGluZDNfZGF0YSA9IGhpbmQyX2RhdGEgJT4lICMgQ29udGFpbnMgZGF0YSBmb3Igc3BlY2llcyB0aGF0IHdvbid0IGNoYW5nZSBmcm9tIHNjZW5hcmlvIDIKICBmaWx0ZXIoIShzcGVjaWVzICVpbiUgY29ycl92YWxzJHNwX25hbWUpKQoKcHJlZHNfdG9fcHVsbCA9IGhpbmRfcHJlZHMgJT4lICAKICBzZWxlY3QoU3BlY2llcywgUHJlZGljdG9yLCBEdXJhdGlvbikgCgpmb3IoaSBpbiAxOmxlbmd0aChwcmVkc190b19wdWxsJFNwZWNpZXMpKXsKICAKICBkdXJhdGlvbiA9IHByZWRzX3RvX3B1bGwkRHVyYXRpb25baV0KICAKICBpZihkdXJhdGlvbiA9PSAwKXsgI1RoZSBwcmlvciBkYXkgdGVtcGVyYXR1cmUgbWV0cmljcyBzaG91bGQgYmUgdXNlZAogICAgCiAgICBwcmVkaWN0b3JzID0gaGluZF9kYWlseV90ZW1wX2RhdGEgJT4lIAogICAgICBtdXRhdGUoZGF0ZSA9IGRhdGUpIAogICAgCiAgICBwYXJhbWV0ZXIgPSAibWVhbl90ZW1wIiAjdXNpbmcgbWVhbiB0ZW1wZXJhdHVyZSBhcyBhIHByb3h5IGZvciBjb2xsZWN0aW9uIHRlbXAKICAgIAogICAgbW9kZWxfZGF0YSA9IGZ1bGxfZGF0YSAlPiUKICAgICAgZmlsdGVyKHNwX25hbWUgJWluJSBwcmVkc190b19wdWxsJFNwZWNpZXNbaV0pICU+JSAKICAgICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogICAgICBtdXRhdGUoY29sbGVjdGlvbl9kYXRlID0gYXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpKSAlPiUgCiAgICAgIGlubmVyX2pvaW4ocHJlZGljdG9ycywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAgCiAgICAgIHNlbGVjdChjdG1heCwgY29udGFpbnMocGFyYW1ldGVyKSkKICAgIAogICAgaWYoZGltKG1vZGVsX2RhdGEpWzJdID09IDIpewogICAgICBoaW5kLm1vZGVsID0gbG0oZGF0YSA9IG1vZGVsX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiAuKQogICAgICAKICAgICAgc3BfZGF0YSA9IHByZWRpY3RvcnMgJT4lIAogICAgICAgIHNlbGVjdChkYXRlLCBjb250YWlucyhwYXJhbWV0ZXIpKSAlPiUgCiAgICAgICAgbXV0YXRlKHByZWRfY3RtYXggPSBwcmVkaWN0KGhpbmQubW9kZWwsIG5ld2RhdGEgPSAuKSkgJT4lICAKICAgICAgICBzZWxlY3QoZGF0ZSwgcHJlZF9jdG1heCkgJT4lIAogICAgICAgIGlubmVyX2pvaW4oaGluZF9kYWlseV90ZW1wX2RhdGEsIGJ5ID0gYygiZGF0ZSIpKSAlPiUgCiAgICAgICAgbXV0YXRlKCJzcGVjaWVzIiA9IHByZWRzX3RvX3B1bGwkU3BlY2llc1tpXSwKICAgICAgICAgICAgICAgImRveSIgPSB5ZGF5KGRhdGUpLAogICAgICAgICAgICAgICBsaW1fZGlmZiA9IHByZWRfY3RtYXggLSBtYXhfdGVtcCkgJT4lIAogICAgICAgIHNlbGVjdChkYXRlLCBkYWlseV9tZWFuID0gbWVhbl90ZW1wLCBkYWlseV9tYXggPSBtYXhfdGVtcCwgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIGRveSkKICAgICAgCiAgICAgIGhpbmQzX2RhdGEgPSBiaW5kX3Jvd3MoaGluZDNfZGF0YSwgc3BfZGF0YSkKICAgIH1lbHNlewogICAgICBwcmludChjKHVuaXF1ZShzcF9kYXRhJHNwZWNpZXMpLCAiVG9vIG1hbnkgY29sdW1ucyBzZWxlY3RlZCIpKQogICAgfQogICAgCiAgfQogIAogIGlmKGR1cmF0aW9uID4gMCl7CiAgICAjTmVpdGhlciB0aGUgcHJpb3IgZGF5IG5vciBkYXkgb2YgbWV0cmljcyBzaG91bGQgYmUgdXNlZDsgdXNlIGR1cmF0aW9uIGFzIG5fZGF5cwogICAgCiAgICBwcmVkaWN0b3JzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gaGluZF9kYWlseV90ZW1wX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gaGluZF90ZW1wX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5fZGF5cyA9IGR1cmF0aW9uKQogICAgCiAgICBwYXJhbWV0ZXIgPSBwcmVkc190b19wdWxsJFByZWRpY3RvcltpXQogICAgCiAgICBtb2RlbF9kYXRhID0gZnVsbF9kYXRhICU+JQogICAgICBmaWx0ZXIoc3BfbmFtZSAlaW4lIHByZWRzX3RvX3B1bGwkU3BlY2llc1tpXSkgJT4lIAogICAgICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiAgICAgIG11dGF0ZShjb2xsZWN0aW9uX2RhdGUgPSBhc19kYXRlKGNvbGxlY3Rpb25fZGF0ZSkpICU+JSAKICAgICAgbGVmdF9qb2luKHByZWRpY3RvcnMsIGpvaW5fYnkoY29sbGVjdGlvbl9kYXRlID09IGRhdGUpKSAlPiUgIAogICAgICBzZWxlY3QoY3RtYXgsIGNvbnRhaW5zKHBhc3RlKCJkYXlfIiwgcGFyYW1ldGVyLCBzZXAgPSAiIikpKQogICAgCiAgICBpZihkaW0obW9kZWxfZGF0YSlbMl0gPT0gMil7CiAgICAgIGhpbmQubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICBjdG1heCB+IC4pCiAgICAgIAogICAgICBzcF9kYXRhID0gcHJlZGljdG9ycyAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGNvbnRhaW5zKHBhcmFtZXRlcikpICU+JSAKICAgICAgICBtdXRhdGUocHJlZF9jdG1heCA9IHByZWRpY3QoaGluZC5tb2RlbCwgbmV3ZGF0YSA9IC4pKSAlPiUgIAogICAgICAgIHNlbGVjdChkYXRlLCBwcmVkX2N0bWF4KSAlPiUgCiAgICAgICAgaW5uZXJfam9pbihoaW5kX2RhaWx5X3RlbXBfZGF0YSwgYnkgPSBjKCJkYXRlIikpICU+JSAKICAgICAgICBtdXRhdGUoInNwZWNpZXMiID0gcHJlZHNfdG9fcHVsbCRTcGVjaWVzW2ldLAogICAgICAgICAgICAgICAiZG95IiA9IHlkYXkoZGF0ZSksCiAgICAgICAgICAgICAgIGxpbV9kaWZmID0gcHJlZF9jdG1heCAtIG1heF90ZW1wKSAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGRhaWx5X21lYW4gPSBtZWFuX3RlbXAsIGRhaWx5X21heCA9IG1heF90ZW1wLCBzcGVjaWVzLCBwcmVkX2N0bWF4LCBsaW1fZGlmZiwgZG95KQogICAgICAKICAgICAgaGluZDNfZGF0YSA9IGJpbmRfcm93cyhoaW5kM19kYXRhLCBzcF9kYXRhKQogICAgICAKICAgIH1lbHNlewogICAgICBwcmludChjKHVuaXF1ZShzcF9kYXRhJHNwZWNpZXMpLCAiVG9vIG1hbnkgY29sdW1ucyBzZWxlY3RlZCIpKQogICAgfQogICAgCiAgfQp9CgoKaGluZDNfZGF0YSA9IGhpbmQzX2RhdGEgJT4lIAogIG11dGF0ZSgibWV0aG9kIiA9ICJWYXJpYWJsZV9hY2NsaW1hdGlvbiIpCmBgYAoKVGhpcyB0aGlyZCBhcHByb2FjaCBkaWQgbm90IGFmZmVjdCB0aGUgcHJlZGljdGVkIHBhdHRlcm5zIGluICpMaW1ub2NhbGFudXMqIG9yICpTZW5lY2VsbGEqIChuZWl0aGVyIHNwZWNpZXMgaGFzIGJlZW4gb2JzZXJ2ZWQgaW4gZW5vdWdoIGNvbGxlY3Rpb25zIHRvIGVzdGltYXRlIHRoZSBlZmZlY3RzIG9mIGRpZmZlcmVudCBlbnZpcm9ubWVudGFsIGZhY3RvcnMpLiBDaGFuZ2luZyB0aGUgYWNjbGltYXRpb24gYXBwcm9hY2ggZGlkIGFmZmVjdCBwYXR0ZXJucyBpbiB0aGVybWFsIGxpbWl0cyBpbiB0aGUgb3RoZXIgc3BlY2llcyB0aG91Z2guIFRoZSBmaWd1cmUgYmVsb3cgc2hvd3MgaG93IHByZWRpY3RlZCB3YXJtaW5nIHRvbGVyYW5jZSB2YXJpZXMgb3ZlciB0aGUgeWVhciBpbiB0aGUgc2V2ZW4gc3BlY2llcywgYmFzZWQgb24gdGhlIHRocmVlIGRpZmZlcmVudCBwcmVkaWN0aW9uIG1ldGhvZHMuIEluIGdlbmVyYWwsIGNvbnN0YW50IHRoZXJtYWwgbGltaXRzICh0aGUgJ25vIGFjY2xpbWF0aW9uJyBtZXRob2QpIHJlc3VsdGVkIGluIGxhcmdlciB3YXJtaW5nIHRvbGVyYW5jZSBkdXJpbmcgdGhlIHdpbnRlciBhbmQgbG93ZXIgd2FybWluZyB0b2xlcmFuY2UgZHVyaW5nIHRoZSBzdW1tZXIsIGFsdGhvdWdoIHRoaXMgZWZmZWN0IHdhcyBzbWFsbCBpbiBtb3N0IHNwZWNpZXMuICAgICAKYGBge3Igc3VwcC1maWctaGluZF9jYXN0X3N1bW1hcnksIGZpZy53aWR0aD0xMywgZmlnLmhlaWdodD0xMH0Kc3ludGhlc2lzID0gYmluZF9yb3dzKAogIHNlbGVjdChoaW5kMV9kYXRhLCBkYXRlLCBkb3ksIGRhaWx5X21lYW4sIGRhaWx5X21heCwgc3BlY2llcywgInByZWRfY3RtYXgiID0gbWVhbl9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCksCiAgc2VsZWN0KGhpbmQyX2RhdGEsIGRhdGUsIGRveSwgZGFpbHlfbWVhbiwgZGFpbHlfbWF4LCAgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCksCiAgc2VsZWN0KGhpbmQzX2RhdGEsIGRhdGUsIGRveSwgZGFpbHlfbWVhbiwgZGFpbHlfbWF4LCAgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIG1ldGhvZCkpICU+JSAKICBtdXRhdGUobWV0aG9kID0gZmN0X3JlbGV2ZWwobWV0aG9kLCAiTm9fYWNjbGltYXRpb24iLCAiQ29uc3RhbnRfYWNjbGltYXRpb24iLCAiVmFyaWFibGVfYWNjbGltYXRpb24iKSkgJT4lIAogIGZpbHRlcighKHNwZWNpZXMgPT0gIk9zcGhyYW50aWN1bSBsYWJyb25lY3R1bSIgJiBtZXRob2QgPT0gIlZhcmlhYmxlX2FjY2xpbWF0aW9uIikpCgpjbGltYXRvbG9neSA9IHN5bnRoZXNpcyAlPiUgCiAgZ3JvdXBfYnkoc3BlY2llcywgZG95LCBtZXRob2QpICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2RpZmYiID0gbWVhbihsaW1fZGlmZiksCiAgICAgICAgICAgICJtaW5fZGlmZiIgPSBtaW4obGltX2RpZmYpLAogICAgICAgICAgICAibWF4X2RpZmYiID0gbWF4KGxpbV9kaWZmKSkgJT4lIAogIG11dGF0ZShtZXRob2QgPSBmY3RfcmVsZXZlbChtZXRob2QsICJOb19hY2NsaW1hdGlvbiIsICJDb25zdGFudF9hY2NsaW1hdGlvbiIsICJWYXJpYWJsZV9hY2NsaW1hdGlvbiIpKQoKYWNjX2VmZmVjdHMgPSBzeW50aGVzaXMgJT4lIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSBjKGRhdGUsIHNwZWNpZXMsIGRveSksIAogICAgICAgICAgICAgIG5hbWVzX2Zyb20gPSBtZXRob2QsIAogICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbGltX2RpZmYpICU+JSAgCiAgbXV0YXRlKCJjb25zdF9hY2NfZWZmZWN0IiA9IENvbnN0YW50X2FjY2xpbWF0aW9uIC0gTm9fYWNjbGltYXRpb24sCiAgICAgICAgICJ2YXJfYWNjX2VmZmVjdCIgPSBWYXJpYWJsZV9hY2NsaW1hdGlvbiAtIE5vX2FjY2xpbWF0aW9uKQoKZ2dwbG90KHN5bnRoZXNpcywgYWVzKHggPSBkb3ksIHkgPSBsaW1fZGlmZiwgY29sb3VyID0gbWV0aG9kKSkgKyAKICBmYWNldF93cmFwKHNwZWNpZXN+LikgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAwKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDUsIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC4xKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgWWVhciIsIAogICAgICAgeSA9ICJQcmVkaWN0ZWQgV2FybWluZyBUb2xlcmFuY2UgKMKwQyBBYm92ZSBEYWlseSBNYXgpIikgKyAKICBndWlkZXMoY29sb3VyID0gZ3VpZGVfbGVnZW5kKG92ZXJyaWRlLmFlcyA9IGxpc3QoYWxwaGEgPSAxKSkpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoYmFzZV9zaXplID0gMTgpICsgCiAgdGhlbWUoc3RyaXAudGV4dC54LnRvcCA9IGVsZW1lbnRfdGV4dChzaXplID0gMTApKQpgYGAKCmBgYHtyIHN1cHAtZmlnLWFjY19zY2VuYXJpb19lZmZlY3RzLCBmaWcud2lkdGg9OSwgZmlnLmhlaWdodD03fQp3dF9oaW5kY2FzdF9zdW1tYXJ5ID0gc3ludGhlc2lzICU+JSAgCiAgbXV0YXRlKCJ5ZWFyIiA9IHllYXIoZGF0ZSkpICU+JSAKICBncm91cF9ieShzcGVjaWVzLCB5ZWFyLCBtZXRob2QpICU+JSAKICBzdW1tYXJpc2UoIm1pbl93dCIgPSBtaW4obGltX2RpZmYpLAogICAgICAgICAgICAibWF4X3d0IiA9IG1heChsaW1fZGlmZikpICU+JSAKICBwaXZvdF9sb25nZXIoY29scyA9IGMobWluX3d0LCBtYXhfd3QpLCAKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAibWV0cmljIiwgCiAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ3dCIpICU+JSAKICBncm91cF9ieShzcGVjaWVzLCBtZXRob2QsIG1ldHJpYykgJT4lIAogIHN1bW1hcmlzZSgibWVhbl93dCIgPSBtZWFuKHd0KSkKCnd0X2hpbmRjYXN0X3N1bW1hcnkgJT4lIAogIGZpbHRlcihtZXRyaWMgPT0gIm1pbl93dCIpICU+JSAKZ2dwbG90KGFlcyh4ID0gbWV0aG9kLCB5ID0gbWVhbl93dCwgZ3JvdXAgPSBzcGVjaWVzLCBjb2xvdXIgPSBzcGVjaWVzKSkgKyAKICAjZmFjZXRfd3JhcCgufm1ldHJpYywgc2NhbGVzID0gImZyZWVfeSIpICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMCkgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSA1KSArIAogIGdlb21fbGluZShsaW5ld2lkdGggPSAyKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDMpICsgCiAgbGFicyh4ID0gIlNjZW5hcmlvIiwgCiAgICAgICB5ID0gIk1lYW4gWWVhcmx5IE1pbmltdW0gV1QgKMKwQykiKSArIAogIHNjYWxlX2NvbG9yX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMzAwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKYGBgCg==